├── .DS_Store
├── .circleci
└── config.yml
├── .github
├── CODEOWNERS
├── ISSUE_TEMPLATE
│ ├── Bug Report.yml
│ ├── Feature Request.yml
│ └── config.yml
├── dependabot.yml
├── stale.yml
└── workflows
│ └── semgrep.yml
├── .gitignore
├── 01-Login
├── .dockerignore
├── .gitignore
├── Dockerfile
├── README.md
├── build.gradle
├── exec.ps1
├── exec.sh
├── gradle
│ └── wrapper
│ │ ├── gradle-wrapper.jar
│ │ └── gradle-wrapper.properties
├── gradlew
├── gradlew.bat
├── settings.gradle
└── src
│ └── main
│ ├── java
│ └── com
│ │ └── auth0
│ │ └── example
│ │ ├── Auth0Filter.java
│ │ ├── AuthenticationControllerProvider.java
│ │ ├── CallbackServlet.java
│ │ ├── HomeServlet.java
│ │ ├── LoginServlet.java
│ │ └── LogoutServlet.java
│ └── webapp
│ ├── WEB-INF
│ ├── jsp
│ │ └── home.jsp
│ └── web.xml
│ ├── css
│ ├── bootstrap-theme.css
│ ├── bootstrap.css
│ ├── home.css
│ ├── jquery.growl.css
│ ├── jumbotron-narrow.css
│ ├── password.css
│ └── signin.css
│ ├── fonts
│ ├── glyphicons-halflings-regular.eot
│ ├── glyphicons-halflings-regular.svg
│ ├── glyphicons-halflings-regular.ttf
│ ├── glyphicons-halflings-regular.woff
│ └── glyphicons-halflings-regular.woff2
│ └── js
│ ├── bootstrap.js
│ └── jquery.growl.js
├── LICENSE
└── README.md
/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/auth0-samples/auth0-servlet-sample/932f880a72c65d7cc4937fd2c4d60d89f306c938/.DS_Store
--------------------------------------------------------------------------------
/.circleci/config.yml:
--------------------------------------------------------------------------------
1 | # Common logic
2 | defaults: &defaults
3 | steps:
4 | - attach_workspace:
5 | at: ~/
6 | - run:
7 | name: Replace Auth0 test credentials
8 | command: |
9 | sed -i 's/{CLIENT_ID}/'$AUTH0_TEST_CLIENT_ID'/g' $AUTH0_CFG
10 | sed -i 's/{DOMAIN}/'$AUTH0_TEST_DOMAIN'/g' $AUTH0_CFG
11 | sed -i 's/{CLIENT_SECRET}/'$AUTH0_TEST_CLIENT_SECRET'/g' $AUTH0_CFG
12 | - run:
13 | name: Build pull request
14 | command: |
15 | docker build -t $CIRCLE_JOB ./$SAMPLE_PATH
16 | docker run -d -it -p 3000:3000 --name $CIRCLE_SHA1 $CIRCLE_JOB
17 | background: true
18 | - run:
19 | name: Wait for app to be available
20 | command: |
21 | sleep 80
22 | docker run --network host --rm appropriate/curl --retry 8 --retry-connrefused -v localhost:3000/portal/home
23 | - run:
24 | name: Run tests
25 | command: |
26 | docker create --network host --name tester codeceptjs/codeceptjs codeceptjs run-multiple --all --steps
27 | docker cp $(pwd)/lock_login_test.js tester:/tests/lock_login_test.js
28 | docker cp $(pwd)/codecept.conf.js tester:/tests/codecept.conf.js
29 | docker start -i tester
30 | working_directory: scripts
31 | - run:
32 | name: Copy app container logs
33 | command: |
34 | mkdir -p /tmp/out
35 | docker logs $CIRCLE_SHA1 > /tmp/out/app_logs.log
36 | docker cp tester:/tests/out /tmp/
37 | when: on_fail
38 | - store_artifacts:
39 | path: /tmp/out
40 |
41 | # Jobs and Workflows
42 | version: 2.1
43 | parameters:
44 | machine_image:
45 | type: string
46 | default: ubuntu-2004:202201-02
47 | jobs:
48 | checkout:
49 | machine:
50 | image: << pipeline.parameters.machine_image >>
51 | steps:
52 | - checkout
53 | - run: git clone -b refactor-for-java https://github.com/auth0-samples/spa-quickstarts-tests scripts
54 | - persist_to_workspace:
55 | root: ~/
56 | paths:
57 | - project
58 | - scripts
59 | login:
60 | machine:
61 | image: << pipeline.parameters.machine_image >>
62 | environment:
63 | - AUTH0_CFG: 01-Login/src/main/webapp/WEB-INF/web.xml
64 | - SAMPLE_PATH: 01-Login
65 | <<: *defaults
66 |
67 | workflows:
68 | version: 2
69 | quickstarts_login:
70 | jobs:
71 | - checkout:
72 | context: Quickstart Web App Test
73 | - login:
74 | context: Quickstart Web App Test
75 | requires:
76 | - checkout
77 |
78 |
--------------------------------------------------------------------------------
/.github/CODEOWNERS:
--------------------------------------------------------------------------------
1 | * @auth0-samples/dx-sdks-engineer
2 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/Bug Report.yml:
--------------------------------------------------------------------------------
1 | name: 🐞 Report a bug
2 | description: Have you found a bug or issue? Create a bug report for this sample
3 |
4 | body:
5 | - type: markdown
6 | attributes:
7 | value: |
8 | **Please do not report security vulnerabilities here**. The [Responsible Disclosure Program](https://auth0.com/responsible-disclosure-policy) details the procedure for disclosing security issues.
9 |
10 | - type: checkboxes
11 | id: checklist
12 | attributes:
13 | label: Checklist
14 | options:
15 | - label: I have looked into the [Readme](https://github.com/auth0-samples/auth0-servlet-sample/tree/master/01-Login#readme) and have not found a suitable solution or answer.
16 | required: true
17 | - label: I have searched the [issues](https://github.com/auth0-samples/auth0-servlet-sample/issues) and have not found a suitable solution or answer.
18 | required: true
19 | - label: I have searched the [Auth0 Community](https://community.auth0.com) forums and have not found a suitable solution or answer.
20 | required: true
21 | - label: I agree to the terms within the [Auth0 Code of Conduct](https://github.com/auth0/open-source-template/blob/master/CODE-OF-CONDUCT.md).
22 | required: true
23 |
24 | - type: textarea
25 | id: description
26 | attributes:
27 | label: Description
28 | description: Provide a clear and concise description of the issue, including what you expected to happen.
29 | validations:
30 | required: true
31 |
32 | - type: textarea
33 | id: reproduction
34 | attributes:
35 | label: Reproduction
36 | description: Detail the steps taken to reproduce this error, and whether this issue can be reproduced consistently or if it is intermittent.
37 | placeholder: |
38 | 1. Step 1...
39 | 2. Step 2...
40 | 3. ...
41 | validations:
42 | required: true
43 |
44 | - type: textarea
45 | id: additional-context
46 | attributes:
47 | label: Additional context
48 | description: Any other relevant information you think would be useful.
49 | validations:
50 | required: false
51 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/Feature Request.yml:
--------------------------------------------------------------------------------
1 | name: 🧩 Feature request
2 | description: Suggest an idea or a feature for this sample
3 | labels: ["feature request"]
4 |
5 | body:
6 | - type: checkboxes
7 | id: checklist
8 | attributes:
9 | label: Checklist
10 | options:
11 | - label: I have looked into the [Readme](https://github.com/auth0-samples/auth0-servlet-sample/tree/master/01-Login#readme) and have not found a suitable solution or answer.
12 | required: true
13 | - label: I have searched the [issues](https://github.com/auth0-samples/auth0-servlet-sample/issues) and have not found a suitable solution or answer.
14 | required: true
15 | - label: I have searched the [Auth0 Community](https://community.auth0.com) forums and have not found a suitable solution or answer.
16 | required: true
17 | - label: I agree to the terms within the [Auth0 Code of Conduct](https://github.com/auth0/open-source-template/blob/master/CODE-OF-CONDUCT.md).
18 | required: true
19 |
20 | - type: textarea
21 | id: description
22 | attributes:
23 | label: Describe the problem you'd like to have solved
24 | description: A clear and concise description of what the problem is.
25 | validations:
26 | required: true
27 |
28 | - type: textarea
29 | id: ideal-solution
30 | attributes:
31 | label: Describe the ideal solution
32 | description: A clear and concise description of what you want to happen.
33 | validations:
34 | required: true
35 |
36 | - type: textarea
37 | id: alternatives-and-workarounds
38 | attributes:
39 | label: Alternatives and current workarounds
40 | description: A clear and concise description of any alternatives you've considered or any workarounds that are currently in place.
41 | validations:
42 | required: false
43 |
44 | - type: textarea
45 | id: additional-context
46 | attributes:
47 | label: Additional context
48 | description: Add any other context or screenshots about the feature request here.
49 | validations:
50 | required: false
51 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE/config.yml:
--------------------------------------------------------------------------------
1 | blank_issues_enabled: false
2 | contact_links:
3 | - name: 🤔 Help & Questions
4 | url: https://community.auth0.com
5 | about: Ask general support or usage questions in the Auth0 Community forums.
6 |
--------------------------------------------------------------------------------
/.github/dependabot.yml:
--------------------------------------------------------------------------------
1 | version: 2
2 | updates:
3 | - package-ecosystem: "gradle"
4 | directory: "/01-Login"
5 | schedule:
6 | interval: "daily"
7 | ignore:
8 | - dependency-name: "*"
9 | update-types: ["version-update:semver-major", "version-update:semver-patch"]
10 |
--------------------------------------------------------------------------------
/.github/stale.yml:
--------------------------------------------------------------------------------
1 | # Configuration for probot-stale - https://github.com/probot/stale
2 |
3 | # Number of days of inactivity before an Issue or Pull Request becomes stale
4 | daysUntilStale: 90
5 |
6 | # Number of days of inactivity before an Issue or Pull Request with the stale label is closed.
7 | daysUntilClose: 7
8 |
9 | # Issues or Pull Requests with these labels will never be considered stale. Set to `[]` to disable
10 | exemptLabels: []
11 |
12 | # Set to true to ignore issues with an assignee (defaults to false)
13 | exemptAssignees: true
14 |
15 | # Label to use when marking as stale
16 | staleLabel: closed:stale
17 |
18 | # Comment to post when marking as stale. Set to `false` to disable
19 | markComment: >
20 | This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. If you have not received a response for our team (apologies for the delay) and this is still a blocker, please reply with additional information or just a ping. Thank you for your contribution! 🙇♂️
--------------------------------------------------------------------------------
/.github/workflows/semgrep.yml:
--------------------------------------------------------------------------------
1 | name: Semgrep
2 |
3 | on:
4 | pull_request: {}
5 |
6 | push:
7 | branches: ["master", "main"]
8 |
9 | schedule:
10 | - cron: '30 0 1,15 * *'
11 |
12 | jobs:
13 | semgrep:
14 | name: Scan
15 | runs-on: ubuntu-latest
16 | container:
17 | image: returntocorp/semgrep
18 | # Skip any PR created by dependabot to avoid permission issues
19 | if: (github.actor != 'dependabot[bot]')
20 | steps:
21 | - uses: actions/checkout@v3
22 |
23 | - run: semgrep ci
24 | env:
25 | SEMGREP_APP_TOKEN: ${{ secrets.SEMGREP_APP_TOKEN }}
26 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .gradle
2 | build/
3 | !gradle/wrapper/gradle-wrapper.jar
4 | **/bin/
5 |
6 | ### STS ###
7 | .apt_generated
8 | .classpath
9 | .factorypath
10 | .project
11 | .settings
12 | .springBeans
13 | .sts4-cache
14 |
15 | ### IntelliJ IDEA ###
16 | .idea
17 | *.iws
18 | *.iml
19 | *.ipr
20 | /out/
21 |
22 | ### VS Code ###
23 | .vscode/
24 |
25 | target/
26 | .DS_Store
--------------------------------------------------------------------------------
/01-Login/.dockerignore:
--------------------------------------------------------------------------------
1 | .gitignore
2 | .gradle/
3 | .idea/
4 | build/
5 | exec.sh
6 | exec.ps1
7 | README.md
8 |
--------------------------------------------------------------------------------
/01-Login/.gitignore:
--------------------------------------------------------------------------------
1 |
2 | # Created by https://www.gitignore.io/api/java,intellij+iml
3 |
4 | ### Intellij+iml ###
5 | # Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio and Webstorm
6 | # Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839
7 |
8 | # Tomcat
9 | target
10 |
11 | # User-specific stuff:
12 | .idea/workspace.xml
13 | .idea/tasks.xml
14 |
15 | # Sensitive or high-churn files:
16 | .idea/dataSources/
17 | .idea/dataSources.ids
18 | .idea/dataSources.xml
19 | .idea/dataSources.local.xml
20 | .idea/sqlDataSources.xml
21 | .idea/dynamic.xml
22 | .idea/uiDesigner.xml
23 |
24 | # Gradle:
25 | .idea/gradle.xml
26 | .idea/libraries
27 |
28 | # Mongo Explorer plugin:
29 | .idea/mongoSettings.xml
30 |
31 | ## File-based project format:
32 | *.iws
33 |
34 | ## Plugin-specific files:
35 |
36 | # IntelliJ
37 | /out/
38 |
39 | # mpeltonen/sbt-idea plugin
40 | .idea_modules/
41 |
42 | # JIRA plugin
43 | atlassian-ide-plugin.xml
44 |
45 | # Crashlytics plugin (for Android Studio and IntelliJ)
46 | com_crashlytics_export_strings.xml
47 | crashlytics.properties
48 | crashlytics-build.properties
49 | fabric.properties
50 |
51 | ### Intellij+iml Patch ###
52 | # Reason: https://github.com/joeblau/gitignore.io/issues/186#issuecomment-249601023
53 |
54 | *.iml
55 | modules.xml
56 | .idea/misc.xml
57 | *.ipr
58 |
59 |
60 | ### Java ###
61 | *.class
62 |
63 | # BlueJ files
64 | *.ctxt
65 |
66 | # Mobile Tools for Java (J2ME)
67 | .mtj.tmp/
68 |
69 | # Package Files #
70 | *.jar
71 | *.war
72 | *.ear
73 |
74 | # virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml
75 | hs_err_pid*
76 |
77 | # End of https://www.gitignore.io/api/java,intellij+iml
78 |
79 | ### Gradle ###
80 | .gradle
81 | /build/
82 |
83 | # Ignore Gradle GUI config
84 | gradle-app.setting
85 |
86 | # Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
87 | !gradle-wrapper.jar
88 |
89 | # Cache of project
90 | .gradletasknamecache
91 |
92 | # # Work around https://youtrack.jetbrains.com/issue/IDEA-116898
93 | # gradle/wrapper/gradle-wrapper.properties
94 |
95 | # End of https://www.gitignore.io/api/java,gradle,osx,windows
96 |
97 | ### OSX ###
98 | *.DS_Store
99 | .AppleDouble
100 | .LSOverride
101 |
102 | # Icon must end with two \r
103 | Icon
104 |
105 |
106 | # Thumbnails
107 | ._*
108 |
109 | # Files that might appear in the root of a volume
110 | .DocumentRevisions-V100
111 | .fseventsd
112 | .Spotlight-V100
113 | .TemporaryItems
114 | .Trashes
115 | .VolumeIcon.icns
116 | .com.apple.timemachine.donotpresent
117 |
118 | # Directories potentially created on remote AFP share
119 | .AppleDB
120 | .AppleDesktop
121 | Network Trash Folder
122 | Temporary Items
123 | .apdisk
--------------------------------------------------------------------------------
/01-Login/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM gradle:8.1.1-jdk11
2 |
3 | WORKDIR /home/gradle
4 | ADD . /home/gradle
5 |
6 | CMD ["gradle", "appRun"]
7 |
8 | EXPOSE 3000
9 |
--------------------------------------------------------------------------------
/01-Login/README.md:
--------------------------------------------------------------------------------
1 |
2 | # Auth0 Servlet Sample
3 |
4 | This sample demonstrates how to use Auth0 to perform authentication using the Auth0 Java MVC Commons library in a Java Servlet web application. Download or clone this repository and follow the instructions below to configure and run the application.
5 |
6 | To learn more about the Auth0 Java MVC Commons library, refer to the project's [documentation](https://github.com/auth0/auth0-java-mvc-common/blob/master/README.md).
7 |
8 | ## Prerequisites
9 |
10 | - Java 8 or greater
11 | - An Auth0 account
12 |
13 | ## Configuration
14 |
15 | ### Auth0 Dashboard
16 | 1. On the [Auth0 Dashboard](https://manage.auth0.com/#/clients) create a new Application of type **Regular Web Application**.
17 | 1. On the **Settings** tab of your application, add the URL `http://localhost:3000/callback` to the **Allowed Callback URLs** field.
18 | 1. On the **Settings** tab of your application, add the URL `http://localhost:3000/login` to the **Allowed Logout URLs** field.
19 | 1. Save the changes to your application settings. Don't close this page; you'll need some of the settings when configuring the application below.
20 |
21 | ### Application configuration
22 |
23 | Set the Auth0 Application values from above in the `src/main/webapp/WEB-INF/web.xml` file.
24 |
25 | ```xml
26 |
27 | com.auth0.domain
28 | {YOUR_AUTH0_DOMAIN}
29 |
30 |
31 |
32 | com.auth0.clientId
33 | {YOUR_AUTH0_CLIENT_ID}
34 |
35 |
36 |
37 | com.auth0.clientSecret
38 | {YOUR_AUTH0_CLIENT_SECRET}
39 |
40 | ```
41 |
42 | ### Running the sample
43 |
44 | Open a terminal or command line, navigate to the `01-Login` directory, and run the following command:
45 |
46 | ```bash
47 | ./gradlew clean appRun
48 | ```
49 |
50 | If you are using a Windows environment, run the following command:
51 |
52 | ```bash
53 | gradlew clean appRun
54 | ```
55 |
56 | The server will be accessible on http://localhost:3000/portal/home. After logging in you should see the `token` in the header.
57 |
58 | ### Running the sample with Docker
59 |
60 | In order to run the example with docker you need to have `docker` installed.
61 |
62 | You also need to set the client credentials as explained [previously](#java-application).
63 |
64 | Execute in command line `sh exec.sh` to run the Docker in Linux, or `.\exec.ps1` to run the Docker in Windows.
65 |
66 | ## Issue Reporting
67 |
68 | If you have found a bug or if you have a feature request, please report them at this repository issues section. Please do not report security vulnerabilities on the public GitHub issue tracker. The [Responsible Disclosure Program](https://auth0.com/whitehat) details the procedure for disclosing security issues.
69 |
70 | ## What is Auth0?
71 |
72 | Auth0 helps you to:
73 |
74 | * Add authentication with [multiple authentication sources](https://docs.auth0.com/identityproviders), either social like **Google, Facebook, Microsoft Account, LinkedIn, GitHub, Twitter, Box, Salesforce, amont others**, or enterprise identity systems like **Windows Azure AD, Google Apps, Active Directory, ADFS or any SAML Identity Provider**.
75 | * Add authentication through more traditional **[username/password databases](https://docs.auth0.com/mysql-connection-tutorial)**.
76 | * Add support for **[linking different user accounts](https://docs.auth0.com/link-accounts)** with the same user.
77 | * Support for generating signed [Json Web Tokens](https://docs.auth0.com/jwt) to call your APIs and **flow the user identity** securely.
78 | * Analytics of how, when and where users are logging in.
79 | * Pull data from other sources and add it to the user profile, through [JavaScript rules](https://docs.auth0.com/rules).
80 |
81 | ## Create a free account in Auth0
82 |
83 | 1. Go to [Auth0](https://auth0.com) and click Sign Up.
84 | 2. Use Google, GitHub or Microsoft Account to login.
85 |
86 | ## Issue Reporting
87 |
88 | If you have found a bug or if you have a feature request, please report them at this repository issues section. Please do not report security vulnerabilities on the public GitHub issue tracker. The [Responsible Disclosure Program](https://auth0.com/whitehat) details the procedure for disclosing security issues.
89 |
90 | ## Author
91 |
92 | [Auth0](https://auth0.com)
93 |
94 | ## License
95 |
96 | This project is licensed under the MIT license. See the [LICENSE](LICENSE.txt) file for more info.
97 |
98 |
--------------------------------------------------------------------------------
/01-Login/build.gradle:
--------------------------------------------------------------------------------
1 | plugins {
2 | id "java"
3 | id "war"
4 | id "org.gretty" version "3.1.1"
5 | }
6 |
7 |
8 | group = 'com.auth0'
9 |
10 | compileJava {
11 | sourceCompatibility '1.8'
12 | targetCompatibility '1.8'
13 | }
14 |
15 | gretty {
16 | httpPort = 3000
17 | host = '0.0.0.0'
18 | contextPath = '/'
19 | servletContainer 'tomcat9'
20 | }
21 |
22 | repositories {
23 | mavenCentral()
24 | }
25 |
26 | dependencies {
27 | implementation 'com.auth0:mvc-auth-commons:1.+'
28 | implementation 'javax.servlet:javax.servlet-api:3.1.0'
29 | runtimeOnly 'javax.servlet:jstl:1.2'
30 | }
31 |
--------------------------------------------------------------------------------
/01-Login/exec.ps1:
--------------------------------------------------------------------------------
1 | docker build -t auth0-samples/auth0-servlet-01-login .
2 | docker run -p 3000:3000 -it auth0-samples/auth0-servlet-01-login
3 |
--------------------------------------------------------------------------------
/01-Login/exec.sh:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 | docker build -t auth0-samples/auth0-servlet-01-login .
3 | docker run -p 3000:3000 -it auth0-samples/auth0-servlet-01-login
4 |
--------------------------------------------------------------------------------
/01-Login/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/auth0-samples/auth0-servlet-sample/932f880a72c65d7cc4937fd2c4d60d89f306c938/01-Login/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/01-Login/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | distributionBase=GRADLE_USER_HOME
2 | distributionPath=wrapper/dists
3 | distributionUrl=https\://services.gradle.org/distributions/gradle-8.1.1-bin.zip
4 | zipStoreBase=GRADLE_USER_HOME
5 | zipStorePath=wrapper/dists
6 |
--------------------------------------------------------------------------------
/01-Login/gradlew:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 |
3 | #
4 | # Copyright © 2015-2021 the original authors.
5 | #
6 | # Licensed under the Apache License, Version 2.0 (the "License");
7 | # you may not use this file except in compliance with the License.
8 | # You may obtain a copy of the License at
9 | #
10 | # https://www.apache.org/licenses/LICENSE-2.0
11 | #
12 | # Unless required by applicable law or agreed to in writing, software
13 | # distributed under the License is distributed on an "AS IS" BASIS,
14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 | # See the License for the specific language governing permissions and
16 | # limitations under the License.
17 | #
18 |
19 | ##############################################################################
20 | #
21 | # Gradle start up script for POSIX generated by Gradle.
22 | #
23 | # Important for running:
24 | #
25 | # (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
26 | # noncompliant, but you have some other compliant shell such as ksh or
27 | # bash, then to run this script, type that shell name before the whole
28 | # command line, like:
29 | #
30 | # ksh Gradle
31 | #
32 | # Busybox and similar reduced shells will NOT work, because this script
33 | # requires all of these POSIX shell features:
34 | # * functions;
35 | # * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
36 | # «${var#prefix}», «${var%suffix}», and «$( cmd )»;
37 | # * compound commands having a testable exit status, especially «case»;
38 | # * various built-in commands including «command», «set», and «ulimit».
39 | #
40 | # Important for patching:
41 | #
42 | # (2) This script targets any POSIX shell, so it avoids extensions provided
43 | # by Bash, Ksh, etc; in particular arrays are avoided.
44 | #
45 | # The "traditional" practice of packing multiple parameters into a
46 | # space-separated string is a well documented source of bugs and security
47 | # problems, so this is (mostly) avoided, by progressively accumulating
48 | # options in "$@", and eventually passing that to Java.
49 | #
50 | # Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
51 | # and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
52 | # see the in-line comments for details.
53 | #
54 | # There are tweaks for specific operating systems such as AIX, CygWin,
55 | # Darwin, MinGW, and NonStop.
56 | #
57 | # (3) This script is generated from the Groovy template
58 | # https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
59 | # within the Gradle project.
60 | #
61 | # You can find Gradle at https://github.com/gradle/gradle/.
62 | #
63 | ##############################################################################
64 |
65 | # Attempt to set APP_HOME
66 |
67 | # Resolve links: $0 may be a link
68 | app_path=$0
69 |
70 | # Need this for daisy-chained symlinks.
71 | while
72 | APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
73 | [ -h "$app_path" ]
74 | do
75 | ls=$( ls -ld "$app_path" )
76 | link=${ls#*' -> '}
77 | case $link in #(
78 | /*) app_path=$link ;; #(
79 | *) app_path=$APP_HOME$link ;;
80 | esac
81 | done
82 |
83 | APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit
84 |
85 | APP_NAME="Gradle"
86 | APP_BASE_NAME=${0##*/}
87 |
88 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
89 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
90 |
91 | # Use the maximum available, or set MAX_FD != -1 to use that value.
92 | MAX_FD=maximum
93 |
94 | warn () {
95 | echo "$*"
96 | } >&2
97 |
98 | die () {
99 | echo
100 | echo "$*"
101 | echo
102 | exit 1
103 | } >&2
104 |
105 | # OS specific support (must be 'true' or 'false').
106 | cygwin=false
107 | msys=false
108 | darwin=false
109 | nonstop=false
110 | case "$( uname )" in #(
111 | CYGWIN* ) cygwin=true ;; #(
112 | Darwin* ) darwin=true ;; #(
113 | MSYS* | MINGW* ) msys=true ;; #(
114 | NONSTOP* ) nonstop=true ;;
115 | esac
116 |
117 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
118 |
119 |
120 | # Determine the Java command to use to start the JVM.
121 | if [ -n "$JAVA_HOME" ] ; then
122 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
123 | # IBM's JDK on AIX uses strange locations for the executables
124 | JAVACMD=$JAVA_HOME/jre/sh/java
125 | else
126 | JAVACMD=$JAVA_HOME/bin/java
127 | fi
128 | if [ ! -x "$JAVACMD" ] ; then
129 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
130 |
131 | Please set the JAVA_HOME variable in your environment to match the
132 | location of your Java installation."
133 | fi
134 | else
135 | JAVACMD=java
136 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
137 |
138 | Please set the JAVA_HOME variable in your environment to match the
139 | location of your Java installation."
140 | fi
141 |
142 | # Increase the maximum file descriptors if we can.
143 | if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
144 | case $MAX_FD in #(
145 | max*)
146 | MAX_FD=$( ulimit -H -n ) ||
147 | warn "Could not query maximum file descriptor limit"
148 | esac
149 | case $MAX_FD in #(
150 | '' | soft) :;; #(
151 | *)
152 | ulimit -n "$MAX_FD" ||
153 | warn "Could not set maximum file descriptor limit to $MAX_FD"
154 | esac
155 | fi
156 |
157 | # Collect all arguments for the java command, stacking in reverse order:
158 | # * args from the command line
159 | # * the main class name
160 | # * -classpath
161 | # * -D...appname settings
162 | # * --module-path (only if needed)
163 | # * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
164 |
165 | # For Cygwin or MSYS, switch paths to Windows format before running java
166 | if "$cygwin" || "$msys" ; then
167 | APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
168 | CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
169 |
170 | JAVACMD=$( cygpath --unix "$JAVACMD" )
171 |
172 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
173 | for arg do
174 | if
175 | case $arg in #(
176 | -*) false ;; # don't mess with options #(
177 | /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
178 | [ -e "$t" ] ;; #(
179 | *) false ;;
180 | esac
181 | then
182 | arg=$( cygpath --path --ignore --mixed "$arg" )
183 | fi
184 | # Roll the args list around exactly as many times as the number of
185 | # args, so each arg winds up back in the position where it started, but
186 | # possibly modified.
187 | #
188 | # NB: a `for` loop captures its iteration list before it begins, so
189 | # changing the positional parameters here affects neither the number of
190 | # iterations, nor the values presented in `arg`.
191 | shift # remove old arg
192 | set -- "$@" "$arg" # push replacement arg
193 | done
194 | fi
195 |
196 | # Collect all arguments for the java command;
197 | # * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of
198 | # shell script including quotes and variable substitutions, so put them in
199 | # double quotes to make sure that they get re-expanded; and
200 | # * put everything else in single quotes, so that it's not re-expanded.
201 |
202 | set -- \
203 | "-Dorg.gradle.appname=$APP_BASE_NAME" \
204 | -classpath "$CLASSPATH" \
205 | org.gradle.wrapper.GradleWrapperMain \
206 | "$@"
207 |
208 | # Use "xargs" to parse quoted args.
209 | #
210 | # With -n1 it outputs one arg per line, with the quotes and backslashes removed.
211 | #
212 | # In Bash we could simply go:
213 | #
214 | # readarray ARGS < <( xargs -n1 <<<"$var" ) &&
215 | # set -- "${ARGS[@]}" "$@"
216 | #
217 | # but POSIX shell has neither arrays nor command substitution, so instead we
218 | # post-process each arg (as a line of input to sed) to backslash-escape any
219 | # character that might be a shell metacharacter, then use eval to reverse
220 | # that process (while maintaining the separation between arguments), and wrap
221 | # the whole thing up as a single "set" statement.
222 | #
223 | # This will of course break if any of these variables contains a newline or
224 | # an unmatched quote.
225 | #
226 |
227 | eval "set -- $(
228 | printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
229 | xargs -n1 |
230 | sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
231 | tr '\n' ' '
232 | )" '"$@"'
233 |
234 | exec "$JAVACMD" "$@"
235 |
--------------------------------------------------------------------------------
/01-Login/gradlew.bat:
--------------------------------------------------------------------------------
1 | @rem
2 | @rem Copyright 2015 the original author or authors.
3 | @rem
4 | @rem Licensed under the Apache License, Version 2.0 (the "License");
5 | @rem you may not use this file except in compliance with the License.
6 | @rem You may obtain a copy of the License at
7 | @rem
8 | @rem https://www.apache.org/licenses/LICENSE-2.0
9 | @rem
10 | @rem Unless required by applicable law or agreed to in writing, software
11 | @rem distributed under the License is distributed on an "AS IS" BASIS,
12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 | @rem See the License for the specific language governing permissions and
14 | @rem limitations under the License.
15 | @rem
16 |
17 | @if "%DEBUG%" == "" @echo off
18 | @rem ##########################################################################
19 | @rem
20 | @rem Gradle startup script for Windows
21 | @rem
22 | @rem ##########################################################################
23 |
24 | @rem Set local scope for the variables with windows NT shell
25 | if "%OS%"=="Windows_NT" setlocal
26 |
27 | set DIRNAME=%~dp0
28 | if "%DIRNAME%" == "" set DIRNAME=.
29 | set APP_BASE_NAME=%~n0
30 | set APP_HOME=%DIRNAME%
31 |
32 | @rem Resolve any "." and ".." in APP_HOME to make it shorter.
33 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
34 |
35 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
36 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
37 |
38 | @rem Find java.exe
39 | if defined JAVA_HOME goto findJavaFromJavaHome
40 |
41 | set JAVA_EXE=java.exe
42 | %JAVA_EXE% -version >NUL 2>&1
43 | if "%ERRORLEVEL%" == "0" goto execute
44 |
45 | echo.
46 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
47 | echo.
48 | echo Please set the JAVA_HOME variable in your environment to match the
49 | echo location of your Java installation.
50 |
51 | goto fail
52 |
53 | :findJavaFromJavaHome
54 | set JAVA_HOME=%JAVA_HOME:"=%
55 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
56 |
57 | if exist "%JAVA_EXE%" goto execute
58 |
59 | echo.
60 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
61 | echo.
62 | echo Please set the JAVA_HOME variable in your environment to match the
63 | echo location of your Java installation.
64 |
65 | goto fail
66 |
67 | :execute
68 | @rem Setup the command line
69 |
70 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
71 |
72 |
73 | @rem Execute Gradle
74 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
75 |
76 | :end
77 | @rem End local scope for the variables with windows NT shell
78 | if "%ERRORLEVEL%"=="0" goto mainEnd
79 |
80 | :fail
81 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
82 | rem the _cmd.exe /c_ return code!
83 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
84 | exit /b 1
85 |
86 | :mainEnd
87 | if "%OS%"=="Windows_NT" endlocal
88 |
89 | :omega
90 |
--------------------------------------------------------------------------------
/01-Login/settings.gradle:
--------------------------------------------------------------------------------
1 | rootProject.name = 'auth0-servlet'
2 |
--------------------------------------------------------------------------------
/01-Login/src/main/java/com/auth0/example/Auth0Filter.java:
--------------------------------------------------------------------------------
1 | package com.auth0.example;
2 |
3 | import com.auth0.SessionUtils;
4 |
5 | import javax.servlet.*;
6 | import javax.servlet.annotation.WebFilter;
7 | import javax.servlet.http.HttpServletRequest;
8 | import javax.servlet.http.HttpServletResponse;
9 | import java.io.IOException;
10 |
11 | /**
12 | * Filter class to check if a valid session exists. This will be true if the User Id is present.
13 | */
14 | @WebFilter(urlPatterns = "/portal/*")
15 | public class Auth0Filter implements Filter {
16 |
17 | @Override
18 | public void init(FilterConfig filterConfig) throws ServletException {
19 | }
20 |
21 | /**
22 | * Perform filter check on this request - verify the User Id is present.
23 | *
24 | * @param request the received request
25 | * @param response the response to send
26 | * @param next the next filter chain
27 | **/
28 | @Override
29 | public void doFilter(ServletRequest request, ServletResponse response, FilterChain next) throws IOException, ServletException {
30 | HttpServletRequest req = (HttpServletRequest) request;
31 | HttpServletResponse res = (HttpServletResponse) response;
32 | String accessToken = (String) SessionUtils.get(req, "accessToken");
33 | String idToken = (String) SessionUtils.get(req, "idToken");
34 | if (accessToken == null && idToken == null) {
35 | res.sendRedirect("/login");
36 | return;
37 | }
38 | next.doFilter(request, response);
39 | }
40 |
41 | @Override
42 | public void destroy() {
43 | }
44 | }
--------------------------------------------------------------------------------
/01-Login/src/main/java/com/auth0/example/AuthenticationControllerProvider.java:
--------------------------------------------------------------------------------
1 | package com.auth0.example;
2 |
3 | import com.auth0.AuthenticationController;
4 | import com.auth0.jwk.JwkProvider;
5 | import com.auth0.jwk.JwkProviderBuilder;
6 |
7 | import javax.servlet.ServletConfig;
8 | import java.io.UnsupportedEncodingException;
9 |
10 | /**
11 | * A class that manages a singleton instance of a {@link JwkProvider} and {@link AuthenticationController} to be used
12 | * by Servlets to authenticate users with Auth0.
13 | *
14 | * Note that each application instance should only create one instance of the {@linkplain AuthenticationController}
15 | * per domain and application to minimize unnecessary resource usage.
16 | */
17 | class AuthenticationControllerProvider {
18 |
19 | private AuthenticationControllerProvider() {}
20 |
21 | private static AuthenticationController INSTANCE;
22 |
23 | // if multiple threads may call this, synchronize this method and consider double locking
24 | static AuthenticationController getInstance(ServletConfig config) throws UnsupportedEncodingException {
25 | if (INSTANCE == null) {
26 | String domain = config.getServletContext().getInitParameter("com.auth0.domain");
27 | String clientId = config.getServletContext().getInitParameter("com.auth0.clientId");
28 | String clientSecret = config.getServletContext().getInitParameter("com.auth0.clientSecret");
29 |
30 | if (domain == null || clientId == null || clientSecret == null) {
31 | throw new IllegalArgumentException("Missing domain, clientId, or clientSecret. Did you update src/main/webapp/WEB-INF/web.xml?");
32 | }
33 |
34 | // JwkProvider required for RS256 tokens. If using HS256, do not use.
35 | JwkProvider jwkProvider = new JwkProviderBuilder(domain).build();
36 | INSTANCE = AuthenticationController.newBuilder(domain, clientId, clientSecret)
37 | .withJwkProvider(jwkProvider)
38 | .build();
39 | }
40 |
41 | return INSTANCE;
42 | }
43 | }
44 |
--------------------------------------------------------------------------------
/01-Login/src/main/java/com/auth0/example/CallbackServlet.java:
--------------------------------------------------------------------------------
1 | package com.auth0.example;
2 |
3 | import com.auth0.AuthenticationController;
4 | import com.auth0.IdentityVerificationException;
5 | import com.auth0.SessionUtils;
6 | import com.auth0.Tokens;
7 |
8 | import javax.servlet.ServletConfig;
9 | import javax.servlet.ServletException;
10 | import javax.servlet.annotation.WebServlet;
11 | import javax.servlet.http.HttpServlet;
12 | import javax.servlet.http.HttpServletRequest;
13 | import javax.servlet.http.HttpServletResponse;
14 | import java.io.IOException;
15 | import java.io.UnsupportedEncodingException;
16 |
17 | /**
18 | * The Servlet endpoint used as the callback handler in the OAuth 2.0 authorization code grant flow.
19 | * It will be called with the authorization code after a successful login.
20 | */
21 | @WebServlet(urlPatterns = {"/callback"})
22 | public class CallbackServlet extends HttpServlet {
23 |
24 | private String redirectOnSuccess;
25 | private String redirectOnFail;
26 | private AuthenticationController authenticationController;
27 |
28 |
29 | /**
30 | * Initialize this servlet with required configuration.
31 | *
32 | * Parameters needed on the Local Servlet scope:
33 | *
34 | * 'com.auth0.redirect_on_success': where to redirect after a successful authentication.
35 | * 'com.auth0.redirect_on_error': where to redirect after a failed authentication.
36 | *
37 | * Parameters needed on the Local/Global Servlet scope:
38 | *
39 | * 'com.auth0.domain': the Auth0 domain.
40 | * 'com.auth0.client_id': the Auth0 Client id.
41 | * 'com.auth0.client_secret': the Auth0 Client secret.
42 | *
43 | */
44 | @Override
45 | public void init(ServletConfig config) throws ServletException {
46 | super.init(config);
47 | redirectOnSuccess = "/portal/home";
48 | redirectOnFail = "/login";
49 |
50 | try {
51 | authenticationController = AuthenticationControllerProvider.getInstance(config);
52 | } catch (UnsupportedEncodingException e) {
53 | throw new ServletException("Couldn't create the AuthenticationController instance. Check the configuration.", e);
54 | }
55 | }
56 |
57 | /**
58 | * Process a call to the redirect_uri with a GET HTTP method.
59 | *
60 | * @param req the received request with the tokens (implicit grant) or the authorization code (code grant) in the parameters.
61 | * @param res the response to send back to the server.
62 | * @throws IOException
63 | * @throws ServletException
64 | */
65 | @Override
66 | public void doGet(HttpServletRequest req, HttpServletResponse res) throws IOException, ServletException {
67 | handle(req, res);
68 | }
69 |
70 |
71 | /**
72 | * Process a call to the redirect_uri with a POST HTTP method. This occurs if the authorize_url included the 'response_mode=form_post' value.
73 | * This is disabled by default. On the Local Servlet scope you can specify the 'com.auth0.allow_post' parameter to enable this behaviour.
74 | *
75 | * @param req the received request with the tokens (implicit grant) or the authorization code (code grant) in the parameters.
76 | * @param res the response to send back to the server.
77 | * @throws IOException
78 | * @throws ServletException
79 | */
80 | @Override
81 | public void doPost(HttpServletRequest req, HttpServletResponse res) throws IOException, ServletException {
82 | handle(req, res);
83 | }
84 |
85 | private void handle(HttpServletRequest req, HttpServletResponse res) throws IOException {
86 | try {
87 | Tokens tokens = authenticationController.handle(req, res);
88 | SessionUtils.set(req, "accessToken", tokens.getAccessToken());
89 | SessionUtils.set(req, "idToken", tokens.getIdToken());
90 | res.sendRedirect(redirectOnSuccess);
91 | } catch (IdentityVerificationException e) {
92 | e.printStackTrace();
93 | res.sendRedirect(redirectOnFail);
94 | }
95 | }
96 |
97 | }
98 |
--------------------------------------------------------------------------------
/01-Login/src/main/java/com/auth0/example/HomeServlet.java:
--------------------------------------------------------------------------------
1 | package com.auth0.example;
2 |
3 | import com.auth0.SessionUtils;
4 |
5 | import javax.servlet.ServletException;
6 | import javax.servlet.annotation.WebServlet;
7 | import javax.servlet.http.HttpServlet;
8 | import javax.servlet.http.HttpServletRequest;
9 | import javax.servlet.http.HttpServletResponse;
10 | import java.io.IOException;
11 |
12 | @WebServlet(urlPatterns = {"/portal/home"})
13 | public class HomeServlet extends HttpServlet {
14 |
15 | @Override
16 | protected void doGet(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {
17 | final String accessToken = (String) SessionUtils.get(req, "accessToken");
18 | final String idToken = (String) SessionUtils.get(req, "idToken");
19 | if (accessToken != null) {
20 | req.setAttribute("userId", accessToken);
21 | } else if (idToken != null) {
22 | req.setAttribute("userId", idToken);
23 | }
24 | req.getRequestDispatcher("/WEB-INF/jsp/home.jsp").forward(req, res);
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/01-Login/src/main/java/com/auth0/example/LoginServlet.java:
--------------------------------------------------------------------------------
1 | package com.auth0.example;
2 |
3 |
4 | import com.auth0.AuthenticationController;
5 |
6 | import javax.servlet.ServletConfig;
7 | import javax.servlet.ServletException;
8 | import javax.servlet.annotation.WebServlet;
9 | import javax.servlet.http.HttpServlet;
10 | import javax.servlet.http.HttpServletRequest;
11 | import javax.servlet.http.HttpServletResponse;
12 | import java.io.IOException;
13 | import java.io.UnsupportedEncodingException;
14 |
15 | @WebServlet(urlPatterns = {"/login"})
16 | public class LoginServlet extends HttpServlet {
17 |
18 | private AuthenticationController authenticationController;
19 | private String domain;
20 |
21 | @Override
22 | public void init(ServletConfig config) throws ServletException {
23 | super.init(config);
24 | domain = config.getServletContext().getInitParameter("com.auth0.domain");
25 | try {
26 | authenticationController = AuthenticationControllerProvider.getInstance(config);
27 | } catch (UnsupportedEncodingException e) {
28 | throw new ServletException("Couldn't create the AuthenticationController instance. Check the configuration.", e);
29 | }
30 | }
31 |
32 | @Override
33 | protected void doGet(final HttpServletRequest req, final HttpServletResponse res) throws ServletException, IOException {
34 | String redirectUri = req.getScheme() + "://" + req.getServerName();
35 | if ((req.getScheme().equals("http") && req.getServerPort() != 80) || (req.getScheme().equals("https") && req.getServerPort() != 443)) {
36 | redirectUri += ":" + req.getServerPort();
37 | }
38 | redirectUri += "/callback";
39 |
40 | String authorizeUrl = authenticationController.buildAuthorizeUrl(req, res, redirectUri)
41 | .build();
42 | res.sendRedirect(authorizeUrl);
43 | }
44 |
45 | }
46 |
--------------------------------------------------------------------------------
/01-Login/src/main/java/com/auth0/example/LogoutServlet.java:
--------------------------------------------------------------------------------
1 | package com.auth0.example;
2 |
3 | import javax.servlet.ServletConfig;
4 | import javax.servlet.ServletException;
5 | import javax.servlet.annotation.WebServlet;
6 | import javax.servlet.http.HttpServlet;
7 | import javax.servlet.http.HttpServletRequest;
8 | import javax.servlet.http.HttpServletResponse;
9 | import java.io.IOException;
10 |
11 | @WebServlet(urlPatterns = {"/logout"})
12 | public class LogoutServlet extends HttpServlet {
13 |
14 | private String domain;
15 | private String clientId;
16 |
17 | @Override
18 | public void init(ServletConfig config) {
19 | domain = config.getServletContext().getInitParameter("com.auth0.domain");
20 | clientId = config.getServletContext().getInitParameter("com.auth0.clientId");
21 | }
22 |
23 | @Override
24 | protected void doGet(final HttpServletRequest request, final HttpServletResponse response) throws ServletException, IOException {
25 | if (request.getSession() != null) {
26 | request.getSession().invalidate();
27 | }
28 | String returnUrl = String.format("%s://%s", request.getScheme(), request.getServerName());
29 | if ((request.getScheme().equals("http") && request.getServerPort() != 80) || (request.getScheme().equals("https") && request.getServerPort() != 443)) {
30 | returnUrl += ":" + request.getServerPort();
31 | }
32 | returnUrl += "/login";
33 | String logoutUrl = String.format(
34 | "https://%s/v2/logout?client_id=%s&returnTo=%s",
35 | domain,
36 | clientId,
37 | returnUrl
38 | );
39 | response.sendRedirect(logoutUrl);
40 | }
41 |
42 | }
43 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/WEB-INF/jsp/home.jsp:
--------------------------------------------------------------------------------
1 | <%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %>
2 | <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
3 |
4 |
5 |
6 |
7 |
8 | Home Page
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
29 |
30 |
Hello ${userId}!
31 |
32 |
33 |
34 |
Subheading
35 |
Donec id elit non mi porta gravida at eget metus. Maecenas faucibus mollis interdum.
36 |
37 |
Subheading
38 |
Morbi leo risus, porta ac consectetur ac, vestibulum at eros. Cras mattis consectetur purus sit amet
39 | fermentum.
40 |
41 |
42 |
43 |
Subheading
44 |
Donec id elit non mi porta gravida at eget metus. Maecenas faucibus mollis interdum.
45 |
46 |
Subheading
47 |
Morbi leo risus, porta ac consectetur ac, vestibulum at eros. Cras mattis consectetur purus sit amet
48 | fermentum.
49 |
50 |
51 |
52 |
55 |
56 |
57 |
58 |
68 |
69 |
70 |
71 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/WEB-INF/web.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
8 | com.auth0.domain
9 | {DOMAIN}
10 |
11 |
12 | com.auth0.clientId
13 | {CLIENT_ID}
14 |
15 |
16 | com.auth0.clientSecret
17 | {CLIENT_SECRET}
18 |
19 |
20 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/css/bootstrap-theme.css:
--------------------------------------------------------------------------------
1 | /*!
2 | * Bootstrap v3.3.6 (http://getbootstrap.com)
3 | * Copyright 2011-2015 Twitter, Inc.
4 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
5 | */
6 | .btn-default,
7 | .btn-primary,
8 | .btn-success,
9 | .btn-info,
10 | .btn-warning,
11 | .btn-danger {
12 | text-shadow: 0 -1px 0 rgba(0, 0, 0, .2);
13 | -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, .15), 0 1px 1px rgba(0, 0, 0, .075);
14 | box-shadow: inset 0 1px 0 rgba(255, 255, 255, .15), 0 1px 1px rgba(0, 0, 0, .075);
15 | }
16 | .btn-default:active,
17 | .btn-primary:active,
18 | .btn-success:active,
19 | .btn-info:active,
20 | .btn-warning:active,
21 | .btn-danger:active,
22 | .btn-default.active,
23 | .btn-primary.active,
24 | .btn-success.active,
25 | .btn-info.active,
26 | .btn-warning.active,
27 | .btn-danger.active {
28 | -webkit-box-shadow: inset 0 3px 5px rgba(0, 0, 0, .125);
29 | box-shadow: inset 0 3px 5px rgba(0, 0, 0, .125);
30 | }
31 | .btn-default.disabled,
32 | .btn-primary.disabled,
33 | .btn-success.disabled,
34 | .btn-info.disabled,
35 | .btn-warning.disabled,
36 | .btn-danger.disabled,
37 | .btn-default[disabled],
38 | .btn-primary[disabled],
39 | .btn-success[disabled],
40 | .btn-info[disabled],
41 | .btn-warning[disabled],
42 | .btn-danger[disabled],
43 | fieldset[disabled] .btn-default,
44 | fieldset[disabled] .btn-primary,
45 | fieldset[disabled] .btn-success,
46 | fieldset[disabled] .btn-info,
47 | fieldset[disabled] .btn-warning,
48 | fieldset[disabled] .btn-danger {
49 | -webkit-box-shadow: none;
50 | box-shadow: none;
51 | }
52 | .btn-default .badge,
53 | .btn-primary .badge,
54 | .btn-success .badge,
55 | .btn-info .badge,
56 | .btn-warning .badge,
57 | .btn-danger .badge {
58 | text-shadow: none;
59 | }
60 | .btn:active,
61 | .btn.active {
62 | background-image: none;
63 | }
64 | .btn-default {
65 | text-shadow: 0 1px 0 #fff;
66 | background-image: -webkit-linear-gradient(top, #fff 0%, #e0e0e0 100%);
67 | background-image: -o-linear-gradient(top, #fff 0%, #e0e0e0 100%);
68 | background-image: -webkit-gradient(linear, left top, left bottom, from(#fff), to(#e0e0e0));
69 | background-image: linear-gradient(to bottom, #fff 0%, #e0e0e0 100%);
70 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#ffe0e0e0', GradientType=0);
71 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
72 | background-repeat: repeat-x;
73 | border-color: #dbdbdb;
74 | border-color: #ccc;
75 | }
76 | .btn-default:hover,
77 | .btn-default:focus {
78 | background-color: #e0e0e0;
79 | background-position: 0 -15px;
80 | }
81 | .btn-default:active,
82 | .btn-default.active {
83 | background-color: #e0e0e0;
84 | border-color: #dbdbdb;
85 | }
86 | .btn-default.disabled,
87 | .btn-default[disabled],
88 | fieldset[disabled] .btn-default,
89 | .btn-default.disabled:hover,
90 | .btn-default[disabled]:hover,
91 | fieldset[disabled] .btn-default:hover,
92 | .btn-default.disabled:focus,
93 | .btn-default[disabled]:focus,
94 | fieldset[disabled] .btn-default:focus,
95 | .btn-default.disabled.focus,
96 | .btn-default[disabled].focus,
97 | fieldset[disabled] .btn-default.focus,
98 | .btn-default.disabled:active,
99 | .btn-default[disabled]:active,
100 | fieldset[disabled] .btn-default:active,
101 | .btn-default.disabled.active,
102 | .btn-default[disabled].active,
103 | fieldset[disabled] .btn-default.active {
104 | background-color: #e0e0e0;
105 | background-image: none;
106 | }
107 | .btn-primary {
108 | background-image: -webkit-linear-gradient(top, #337ab7 0%, #265a88 100%);
109 | background-image: -o-linear-gradient(top, #337ab7 0%, #265a88 100%);
110 | background-image: -webkit-gradient(linear, left top, left bottom, from(#337ab7), to(#265a88));
111 | background-image: linear-gradient(to bottom, #337ab7 0%, #265a88 100%);
112 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff265a88', GradientType=0);
113 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
114 | background-repeat: repeat-x;
115 | border-color: #245580;
116 | }
117 | .btn-primary:hover,
118 | .btn-primary:focus {
119 | background-color: #265a88;
120 | background-position: 0 -15px;
121 | }
122 | .btn-primary:active,
123 | .btn-primary.active {
124 | background-color: #265a88;
125 | border-color: #245580;
126 | }
127 | .btn-primary.disabled,
128 | .btn-primary[disabled],
129 | fieldset[disabled] .btn-primary,
130 | .btn-primary.disabled:hover,
131 | .btn-primary[disabled]:hover,
132 | fieldset[disabled] .btn-primary:hover,
133 | .btn-primary.disabled:focus,
134 | .btn-primary[disabled]:focus,
135 | fieldset[disabled] .btn-primary:focus,
136 | .btn-primary.disabled.focus,
137 | .btn-primary[disabled].focus,
138 | fieldset[disabled] .btn-primary.focus,
139 | .btn-primary.disabled:active,
140 | .btn-primary[disabled]:active,
141 | fieldset[disabled] .btn-primary:active,
142 | .btn-primary.disabled.active,
143 | .btn-primary[disabled].active,
144 | fieldset[disabled] .btn-primary.active {
145 | background-color: #265a88;
146 | background-image: none;
147 | }
148 | .btn-success {
149 | background-image: -webkit-linear-gradient(top, #5cb85c 0%, #419641 100%);
150 | background-image: -o-linear-gradient(top, #5cb85c 0%, #419641 100%);
151 | background-image: -webkit-gradient(linear, left top, left bottom, from(#5cb85c), to(#419641));
152 | background-image: linear-gradient(to bottom, #5cb85c 0%, #419641 100%);
153 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5cb85c', endColorstr='#ff419641', GradientType=0);
154 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
155 | background-repeat: repeat-x;
156 | border-color: #3e8f3e;
157 | }
158 | .btn-success:hover,
159 | .btn-success:focus {
160 | background-color: #419641;
161 | background-position: 0 -15px;
162 | }
163 | .btn-success:active,
164 | .btn-success.active {
165 | background-color: #419641;
166 | border-color: #3e8f3e;
167 | }
168 | .btn-success.disabled,
169 | .btn-success[disabled],
170 | fieldset[disabled] .btn-success,
171 | .btn-success.disabled:hover,
172 | .btn-success[disabled]:hover,
173 | fieldset[disabled] .btn-success:hover,
174 | .btn-success.disabled:focus,
175 | .btn-success[disabled]:focus,
176 | fieldset[disabled] .btn-success:focus,
177 | .btn-success.disabled.focus,
178 | .btn-success[disabled].focus,
179 | fieldset[disabled] .btn-success.focus,
180 | .btn-success.disabled:active,
181 | .btn-success[disabled]:active,
182 | fieldset[disabled] .btn-success:active,
183 | .btn-success.disabled.active,
184 | .btn-success[disabled].active,
185 | fieldset[disabled] .btn-success.active {
186 | background-color: #419641;
187 | background-image: none;
188 | }
189 | .btn-info {
190 | background-image: -webkit-linear-gradient(top, #5bc0de 0%, #2aabd2 100%);
191 | background-image: -o-linear-gradient(top, #5bc0de 0%, #2aabd2 100%);
192 | background-image: -webkit-gradient(linear, left top, left bottom, from(#5bc0de), to(#2aabd2));
193 | background-image: linear-gradient(to bottom, #5bc0de 0%, #2aabd2 100%);
194 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5bc0de', endColorstr='#ff2aabd2', GradientType=0);
195 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
196 | background-repeat: repeat-x;
197 | border-color: #28a4c9;
198 | }
199 | .btn-info:hover,
200 | .btn-info:focus {
201 | background-color: #2aabd2;
202 | background-position: 0 -15px;
203 | }
204 | .btn-info:active,
205 | .btn-info.active {
206 | background-color: #2aabd2;
207 | border-color: #28a4c9;
208 | }
209 | .btn-info.disabled,
210 | .btn-info[disabled],
211 | fieldset[disabled] .btn-info,
212 | .btn-info.disabled:hover,
213 | .btn-info[disabled]:hover,
214 | fieldset[disabled] .btn-info:hover,
215 | .btn-info.disabled:focus,
216 | .btn-info[disabled]:focus,
217 | fieldset[disabled] .btn-info:focus,
218 | .btn-info.disabled.focus,
219 | .btn-info[disabled].focus,
220 | fieldset[disabled] .btn-info.focus,
221 | .btn-info.disabled:active,
222 | .btn-info[disabled]:active,
223 | fieldset[disabled] .btn-info:active,
224 | .btn-info.disabled.active,
225 | .btn-info[disabled].active,
226 | fieldset[disabled] .btn-info.active {
227 | background-color: #2aabd2;
228 | background-image: none;
229 | }
230 | .btn-warning {
231 | background-image: -webkit-linear-gradient(top, #f0ad4e 0%, #eb9316 100%);
232 | background-image: -o-linear-gradient(top, #f0ad4e 0%, #eb9316 100%);
233 | background-image: -webkit-gradient(linear, left top, left bottom, from(#f0ad4e), to(#eb9316));
234 | background-image: linear-gradient(to bottom, #f0ad4e 0%, #eb9316 100%);
235 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff0ad4e', endColorstr='#ffeb9316', GradientType=0);
236 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
237 | background-repeat: repeat-x;
238 | border-color: #e38d13;
239 | }
240 | .btn-warning:hover,
241 | .btn-warning:focus {
242 | background-color: #eb9316;
243 | background-position: 0 -15px;
244 | }
245 | .btn-warning:active,
246 | .btn-warning.active {
247 | background-color: #eb9316;
248 | border-color: #e38d13;
249 | }
250 | .btn-warning.disabled,
251 | .btn-warning[disabled],
252 | fieldset[disabled] .btn-warning,
253 | .btn-warning.disabled:hover,
254 | .btn-warning[disabled]:hover,
255 | fieldset[disabled] .btn-warning:hover,
256 | .btn-warning.disabled:focus,
257 | .btn-warning[disabled]:focus,
258 | fieldset[disabled] .btn-warning:focus,
259 | .btn-warning.disabled.focus,
260 | .btn-warning[disabled].focus,
261 | fieldset[disabled] .btn-warning.focus,
262 | .btn-warning.disabled:active,
263 | .btn-warning[disabled]:active,
264 | fieldset[disabled] .btn-warning:active,
265 | .btn-warning.disabled.active,
266 | .btn-warning[disabled].active,
267 | fieldset[disabled] .btn-warning.active {
268 | background-color: #eb9316;
269 | background-image: none;
270 | }
271 | .btn-danger {
272 | background-image: -webkit-linear-gradient(top, #d9534f 0%, #c12e2a 100%);
273 | background-image: -o-linear-gradient(top, #d9534f 0%, #c12e2a 100%);
274 | background-image: -webkit-gradient(linear, left top, left bottom, from(#d9534f), to(#c12e2a));
275 | background-image: linear-gradient(to bottom, #d9534f 0%, #c12e2a 100%);
276 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9534f', endColorstr='#ffc12e2a', GradientType=0);
277 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
278 | background-repeat: repeat-x;
279 | border-color: #b92c28;
280 | }
281 | .btn-danger:hover,
282 | .btn-danger:focus {
283 | background-color: #c12e2a;
284 | background-position: 0 -15px;
285 | }
286 | .btn-danger:active,
287 | .btn-danger.active {
288 | background-color: #c12e2a;
289 | border-color: #b92c28;
290 | }
291 | .btn-danger.disabled,
292 | .btn-danger[disabled],
293 | fieldset[disabled] .btn-danger,
294 | .btn-danger.disabled:hover,
295 | .btn-danger[disabled]:hover,
296 | fieldset[disabled] .btn-danger:hover,
297 | .btn-danger.disabled:focus,
298 | .btn-danger[disabled]:focus,
299 | fieldset[disabled] .btn-danger:focus,
300 | .btn-danger.disabled.focus,
301 | .btn-danger[disabled].focus,
302 | fieldset[disabled] .btn-danger.focus,
303 | .btn-danger.disabled:active,
304 | .btn-danger[disabled]:active,
305 | fieldset[disabled] .btn-danger:active,
306 | .btn-danger.disabled.active,
307 | .btn-danger[disabled].active,
308 | fieldset[disabled] .btn-danger.active {
309 | background-color: #c12e2a;
310 | background-image: none;
311 | }
312 | .thumbnail,
313 | .img-thumbnail {
314 | -webkit-box-shadow: 0 1px 2px rgba(0, 0, 0, .075);
315 | box-shadow: 0 1px 2px rgba(0, 0, 0, .075);
316 | }
317 | .dropdown-menu > li > a:hover,
318 | .dropdown-menu > li > a:focus {
319 | background-color: #e8e8e8;
320 | background-image: -webkit-linear-gradient(top, #f5f5f5 0%, #e8e8e8 100%);
321 | background-image: -o-linear-gradient(top, #f5f5f5 0%, #e8e8e8 100%);
322 | background-image: -webkit-gradient(linear, left top, left bottom, from(#f5f5f5), to(#e8e8e8));
323 | background-image: linear-gradient(to bottom, #f5f5f5 0%, #e8e8e8 100%);
324 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff5f5f5', endColorstr='#ffe8e8e8', GradientType=0);
325 | background-repeat: repeat-x;
326 | }
327 | .dropdown-menu > .active > a,
328 | .dropdown-menu > .active > a:hover,
329 | .dropdown-menu > .active > a:focus {
330 | background-color: #2e6da4;
331 | background-image: -webkit-linear-gradient(top, #337ab7 0%, #2e6da4 100%);
332 | background-image: -o-linear-gradient(top, #337ab7 0%, #2e6da4 100%);
333 | background-image: -webkit-gradient(linear, left top, left bottom, from(#337ab7), to(#2e6da4));
334 | background-image: linear-gradient(to bottom, #337ab7 0%, #2e6da4 100%);
335 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2e6da4', GradientType=0);
336 | background-repeat: repeat-x;
337 | }
338 | .navbar-default {
339 | background-image: -webkit-linear-gradient(top, #fff 0%, #f8f8f8 100%);
340 | background-image: -o-linear-gradient(top, #fff 0%, #f8f8f8 100%);
341 | background-image: -webkit-gradient(linear, left top, left bottom, from(#fff), to(#f8f8f8));
342 | background-image: linear-gradient(to bottom, #fff 0%, #f8f8f8 100%);
343 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#fff8f8f8', GradientType=0);
344 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
345 | background-repeat: repeat-x;
346 | border-radius: 4px;
347 | -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, .15), 0 1px 5px rgba(0, 0, 0, .075);
348 | box-shadow: inset 0 1px 0 rgba(255, 255, 255, .15), 0 1px 5px rgba(0, 0, 0, .075);
349 | }
350 | .navbar-default .navbar-nav > .open > a,
351 | .navbar-default .navbar-nav > .active > a {
352 | background-image: -webkit-linear-gradient(top, #dbdbdb 0%, #e2e2e2 100%);
353 | background-image: -o-linear-gradient(top, #dbdbdb 0%, #e2e2e2 100%);
354 | background-image: -webkit-gradient(linear, left top, left bottom, from(#dbdbdb), to(#e2e2e2));
355 | background-image: linear-gradient(to bottom, #dbdbdb 0%, #e2e2e2 100%);
356 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdbdbdb', endColorstr='#ffe2e2e2', GradientType=0);
357 | background-repeat: repeat-x;
358 | -webkit-box-shadow: inset 0 3px 9px rgba(0, 0, 0, .075);
359 | box-shadow: inset 0 3px 9px rgba(0, 0, 0, .075);
360 | }
361 | .navbar-brand,
362 | .navbar-nav > li > a {
363 | text-shadow: 0 1px 0 rgba(255, 255, 255, .25);
364 | }
365 | .navbar-inverse {
366 | background-image: -webkit-linear-gradient(top, #3c3c3c 0%, #222 100%);
367 | background-image: -o-linear-gradient(top, #3c3c3c 0%, #222 100%);
368 | background-image: -webkit-gradient(linear, left top, left bottom, from(#3c3c3c), to(#222));
369 | background-image: linear-gradient(to bottom, #3c3c3c 0%, #222 100%);
370 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff3c3c3c', endColorstr='#ff222222', GradientType=0);
371 | filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
372 | background-repeat: repeat-x;
373 | border-radius: 4px;
374 | }
375 | .navbar-inverse .navbar-nav > .open > a,
376 | .navbar-inverse .navbar-nav > .active > a {
377 | background-image: -webkit-linear-gradient(top, #080808 0%, #0f0f0f 100%);
378 | background-image: -o-linear-gradient(top, #080808 0%, #0f0f0f 100%);
379 | background-image: -webkit-gradient(linear, left top, left bottom, from(#080808), to(#0f0f0f));
380 | background-image: linear-gradient(to bottom, #080808 0%, #0f0f0f 100%);
381 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff080808', endColorstr='#ff0f0f0f', GradientType=0);
382 | background-repeat: repeat-x;
383 | -webkit-box-shadow: inset 0 3px 9px rgba(0, 0, 0, .25);
384 | box-shadow: inset 0 3px 9px rgba(0, 0, 0, .25);
385 | }
386 | .navbar-inverse .navbar-brand,
387 | .navbar-inverse .navbar-nav > li > a {
388 | text-shadow: 0 -1px 0 rgba(0, 0, 0, .25);
389 | }
390 | .navbar-static-top,
391 | .navbar-fixed-top,
392 | .navbar-fixed-bottom {
393 | border-radius: 0;
394 | }
395 | @media (max-width: 767px) {
396 | .navbar .navbar-nav .open .dropdown-menu > .active > a,
397 | .navbar .navbar-nav .open .dropdown-menu > .active > a:hover,
398 | .navbar .navbar-nav .open .dropdown-menu > .active > a:focus {
399 | color: #fff;
400 | background-image: -webkit-linear-gradient(top, #337ab7 0%, #2e6da4 100%);
401 | background-image: -o-linear-gradient(top, #337ab7 0%, #2e6da4 100%);
402 | background-image: -webkit-gradient(linear, left top, left bottom, from(#337ab7), to(#2e6da4));
403 | background-image: linear-gradient(to bottom, #337ab7 0%, #2e6da4 100%);
404 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2e6da4', GradientType=0);
405 | background-repeat: repeat-x;
406 | }
407 | }
408 | .alert {
409 | text-shadow: 0 1px 0 rgba(255, 255, 255, .2);
410 | -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, .25), 0 1px 2px rgba(0, 0, 0, .05);
411 | box-shadow: inset 0 1px 0 rgba(255, 255, 255, .25), 0 1px 2px rgba(0, 0, 0, .05);
412 | }
413 | .alert-success {
414 | background-image: -webkit-linear-gradient(top, #dff0d8 0%, #c8e5bc 100%);
415 | background-image: -o-linear-gradient(top, #dff0d8 0%, #c8e5bc 100%);
416 | background-image: -webkit-gradient(linear, left top, left bottom, from(#dff0d8), to(#c8e5bc));
417 | background-image: linear-gradient(to bottom, #dff0d8 0%, #c8e5bc 100%);
418 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdff0d8', endColorstr='#ffc8e5bc', GradientType=0);
419 | background-repeat: repeat-x;
420 | border-color: #b2dba1;
421 | }
422 | .alert-info {
423 | background-image: -webkit-linear-gradient(top, #d9edf7 0%, #b9def0 100%);
424 | background-image: -o-linear-gradient(top, #d9edf7 0%, #b9def0 100%);
425 | background-image: -webkit-gradient(linear, left top, left bottom, from(#d9edf7), to(#b9def0));
426 | background-image: linear-gradient(to bottom, #d9edf7 0%, #b9def0 100%);
427 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9edf7', endColorstr='#ffb9def0', GradientType=0);
428 | background-repeat: repeat-x;
429 | border-color: #9acfea;
430 | }
431 | .alert-warning {
432 | background-image: -webkit-linear-gradient(top, #fcf8e3 0%, #f8efc0 100%);
433 | background-image: -o-linear-gradient(top, #fcf8e3 0%, #f8efc0 100%);
434 | background-image: -webkit-gradient(linear, left top, left bottom, from(#fcf8e3), to(#f8efc0));
435 | background-image: linear-gradient(to bottom, #fcf8e3 0%, #f8efc0 100%);
436 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fffcf8e3', endColorstr='#fff8efc0', GradientType=0);
437 | background-repeat: repeat-x;
438 | border-color: #f5e79e;
439 | }
440 | .alert-danger {
441 | background-image: -webkit-linear-gradient(top, #f2dede 0%, #e7c3c3 100%);
442 | background-image: -o-linear-gradient(top, #f2dede 0%, #e7c3c3 100%);
443 | background-image: -webkit-gradient(linear, left top, left bottom, from(#f2dede), to(#e7c3c3));
444 | background-image: linear-gradient(to bottom, #f2dede 0%, #e7c3c3 100%);
445 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff2dede', endColorstr='#ffe7c3c3', GradientType=0);
446 | background-repeat: repeat-x;
447 | border-color: #dca7a7;
448 | }
449 | .progress {
450 | background-image: -webkit-linear-gradient(top, #ebebeb 0%, #f5f5f5 100%);
451 | background-image: -o-linear-gradient(top, #ebebeb 0%, #f5f5f5 100%);
452 | background-image: -webkit-gradient(linear, left top, left bottom, from(#ebebeb), to(#f5f5f5));
453 | background-image: linear-gradient(to bottom, #ebebeb 0%, #f5f5f5 100%);
454 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffebebeb', endColorstr='#fff5f5f5', GradientType=0);
455 | background-repeat: repeat-x;
456 | }
457 | .progress-bar {
458 | background-image: -webkit-linear-gradient(top, #337ab7 0%, #286090 100%);
459 | background-image: -o-linear-gradient(top, #337ab7 0%, #286090 100%);
460 | background-image: -webkit-gradient(linear, left top, left bottom, from(#337ab7), to(#286090));
461 | background-image: linear-gradient(to bottom, #337ab7 0%, #286090 100%);
462 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff286090', GradientType=0);
463 | background-repeat: repeat-x;
464 | }
465 | .progress-bar-success {
466 | background-image: -webkit-linear-gradient(top, #5cb85c 0%, #449d44 100%);
467 | background-image: -o-linear-gradient(top, #5cb85c 0%, #449d44 100%);
468 | background-image: -webkit-gradient(linear, left top, left bottom, from(#5cb85c), to(#449d44));
469 | background-image: linear-gradient(to bottom, #5cb85c 0%, #449d44 100%);
470 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5cb85c', endColorstr='#ff449d44', GradientType=0);
471 | background-repeat: repeat-x;
472 | }
473 | .progress-bar-info {
474 | background-image: -webkit-linear-gradient(top, #5bc0de 0%, #31b0d5 100%);
475 | background-image: -o-linear-gradient(top, #5bc0de 0%, #31b0d5 100%);
476 | background-image: -webkit-gradient(linear, left top, left bottom, from(#5bc0de), to(#31b0d5));
477 | background-image: linear-gradient(to bottom, #5bc0de 0%, #31b0d5 100%);
478 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5bc0de', endColorstr='#ff31b0d5', GradientType=0);
479 | background-repeat: repeat-x;
480 | }
481 | .progress-bar-warning {
482 | background-image: -webkit-linear-gradient(top, #f0ad4e 0%, #ec971f 100%);
483 | background-image: -o-linear-gradient(top, #f0ad4e 0%, #ec971f 100%);
484 | background-image: -webkit-gradient(linear, left top, left bottom, from(#f0ad4e), to(#ec971f));
485 | background-image: linear-gradient(to bottom, #f0ad4e 0%, #ec971f 100%);
486 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff0ad4e', endColorstr='#ffec971f', GradientType=0);
487 | background-repeat: repeat-x;
488 | }
489 | .progress-bar-danger {
490 | background-image: -webkit-linear-gradient(top, #d9534f 0%, #c9302c 100%);
491 | background-image: -o-linear-gradient(top, #d9534f 0%, #c9302c 100%);
492 | background-image: -webkit-gradient(linear, left top, left bottom, from(#d9534f), to(#c9302c));
493 | background-image: linear-gradient(to bottom, #d9534f 0%, #c9302c 100%);
494 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9534f', endColorstr='#ffc9302c', GradientType=0);
495 | background-repeat: repeat-x;
496 | }
497 | .progress-bar-striped {
498 | background-image: -webkit-linear-gradient(45deg, rgba(255, 255, 255, .15) 25%, transparent 25%, transparent 50%, rgba(255, 255, 255, .15) 50%, rgba(255, 255, 255, .15) 75%, transparent 75%, transparent);
499 | background-image: -o-linear-gradient(45deg, rgba(255, 255, 255, .15) 25%, transparent 25%, transparent 50%, rgba(255, 255, 255, .15) 50%, rgba(255, 255, 255, .15) 75%, transparent 75%, transparent);
500 | background-image: linear-gradient(45deg, rgba(255, 255, 255, .15) 25%, transparent 25%, transparent 50%, rgba(255, 255, 255, .15) 50%, rgba(255, 255, 255, .15) 75%, transparent 75%, transparent);
501 | }
502 | .list-group {
503 | border-radius: 4px;
504 | -webkit-box-shadow: 0 1px 2px rgba(0, 0, 0, .075);
505 | box-shadow: 0 1px 2px rgba(0, 0, 0, .075);
506 | }
507 | .list-group-item.active,
508 | .list-group-item.active:hover,
509 | .list-group-item.active:focus {
510 | text-shadow: 0 -1px 0 #286090;
511 | background-image: -webkit-linear-gradient(top, #337ab7 0%, #2b669a 100%);
512 | background-image: -o-linear-gradient(top, #337ab7 0%, #2b669a 100%);
513 | background-image: -webkit-gradient(linear, left top, left bottom, from(#337ab7), to(#2b669a));
514 | background-image: linear-gradient(to bottom, #337ab7 0%, #2b669a 100%);
515 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2b669a', GradientType=0);
516 | background-repeat: repeat-x;
517 | border-color: #2b669a;
518 | }
519 | .list-group-item.active .badge,
520 | .list-group-item.active:hover .badge,
521 | .list-group-item.active:focus .badge {
522 | text-shadow: none;
523 | }
524 | .panel {
525 | -webkit-box-shadow: 0 1px 2px rgba(0, 0, 0, .05);
526 | box-shadow: 0 1px 2px rgba(0, 0, 0, .05);
527 | }
528 | .panel-default > .panel-heading {
529 | background-image: -webkit-linear-gradient(top, #f5f5f5 0%, #e8e8e8 100%);
530 | background-image: -o-linear-gradient(top, #f5f5f5 0%, #e8e8e8 100%);
531 | background-image: -webkit-gradient(linear, left top, left bottom, from(#f5f5f5), to(#e8e8e8));
532 | background-image: linear-gradient(to bottom, #f5f5f5 0%, #e8e8e8 100%);
533 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff5f5f5', endColorstr='#ffe8e8e8', GradientType=0);
534 | background-repeat: repeat-x;
535 | }
536 | .panel-primary > .panel-heading {
537 | background-image: -webkit-linear-gradient(top, #337ab7 0%, #2e6da4 100%);
538 | background-image: -o-linear-gradient(top, #337ab7 0%, #2e6da4 100%);
539 | background-image: -webkit-gradient(linear, left top, left bottom, from(#337ab7), to(#2e6da4));
540 | background-image: linear-gradient(to bottom, #337ab7 0%, #2e6da4 100%);
541 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2e6da4', GradientType=0);
542 | background-repeat: repeat-x;
543 | }
544 | .panel-success > .panel-heading {
545 | background-image: -webkit-linear-gradient(top, #dff0d8 0%, #d0e9c6 100%);
546 | background-image: -o-linear-gradient(top, #dff0d8 0%, #d0e9c6 100%);
547 | background-image: -webkit-gradient(linear, left top, left bottom, from(#dff0d8), to(#d0e9c6));
548 | background-image: linear-gradient(to bottom, #dff0d8 0%, #d0e9c6 100%);
549 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdff0d8', endColorstr='#ffd0e9c6', GradientType=0);
550 | background-repeat: repeat-x;
551 | }
552 | .panel-info > .panel-heading {
553 | background-image: -webkit-linear-gradient(top, #d9edf7 0%, #c4e3f3 100%);
554 | background-image: -o-linear-gradient(top, #d9edf7 0%, #c4e3f3 100%);
555 | background-image: -webkit-gradient(linear, left top, left bottom, from(#d9edf7), to(#c4e3f3));
556 | background-image: linear-gradient(to bottom, #d9edf7 0%, #c4e3f3 100%);
557 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9edf7', endColorstr='#ffc4e3f3', GradientType=0);
558 | background-repeat: repeat-x;
559 | }
560 | .panel-warning > .panel-heading {
561 | background-image: -webkit-linear-gradient(top, #fcf8e3 0%, #faf2cc 100%);
562 | background-image: -o-linear-gradient(top, #fcf8e3 0%, #faf2cc 100%);
563 | background-image: -webkit-gradient(linear, left top, left bottom, from(#fcf8e3), to(#faf2cc));
564 | background-image: linear-gradient(to bottom, #fcf8e3 0%, #faf2cc 100%);
565 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fffcf8e3', endColorstr='#fffaf2cc', GradientType=0);
566 | background-repeat: repeat-x;
567 | }
568 | .panel-danger > .panel-heading {
569 | background-image: -webkit-linear-gradient(top, #f2dede 0%, #ebcccc 100%);
570 | background-image: -o-linear-gradient(top, #f2dede 0%, #ebcccc 100%);
571 | background-image: -webkit-gradient(linear, left top, left bottom, from(#f2dede), to(#ebcccc));
572 | background-image: linear-gradient(to bottom, #f2dede 0%, #ebcccc 100%);
573 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff2dede', endColorstr='#ffebcccc', GradientType=0);
574 | background-repeat: repeat-x;
575 | }
576 | .well {
577 | background-image: -webkit-linear-gradient(top, #e8e8e8 0%, #f5f5f5 100%);
578 | background-image: -o-linear-gradient(top, #e8e8e8 0%, #f5f5f5 100%);
579 | background-image: -webkit-gradient(linear, left top, left bottom, from(#e8e8e8), to(#f5f5f5));
580 | background-image: linear-gradient(to bottom, #e8e8e8 0%, #f5f5f5 100%);
581 | filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffe8e8e8', endColorstr='#fff5f5f5', GradientType=0);
582 | background-repeat: repeat-x;
583 | border-color: #dcdcdc;
584 | -webkit-box-shadow: inset 0 1px 3px rgba(0, 0, 0, .05), 0 1px 0 rgba(255, 255, 255, .1);
585 | box-shadow: inset 0 1px 3px rgba(0, 0, 0, .05), 0 1px 0 rgba(255, 255, 255, .1);
586 | }
587 | /*# sourceMappingURL=bootstrap-theme.css.map */
588 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/css/home.css:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | #password-login > a {
5 | color: red;
6 | }
7 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/css/jquery.growl.css:
--------------------------------------------------------------------------------
1 | /* jQuery Growl
2 | * Copyright 2015 Kevin Sylvestre
3 | * 1.3.1
4 | */
5 | #growls {
6 | z-index: 50000;
7 | position: fixed; }
8 | #growls.default {
9 | top: 10px;
10 | right: 10px; }
11 | #growls.tl {
12 | top: 10px;
13 | left: 10px; }
14 | #growls.tr {
15 | top: 10px;
16 | right: 10px; }
17 | #growls.bl {
18 | bottom: 10px;
19 | left: 10px; }
20 | #growls.br {
21 | bottom: 10px;
22 | right: 10px; }
23 | #growls.tc {
24 | top: 10px;
25 | right: 10px;
26 | left: 10px; }
27 | #growls.bc {
28 | bottom: 10px;
29 | right: 10px;
30 | left: 10px; }
31 | #growls.tc .growl, #growls.bc .growl {
32 | margin-left: auto;
33 | margin-right: auto; }
34 |
35 | .growl {
36 | opacity: 0.8;
37 | filter: alpha(opacity=80);
38 | position: relative;
39 | border-radius: 4px;
40 | -webkit-transition: all 0.4s ease-in-out;
41 | -moz-transition: all 0.4s ease-in-out;
42 | transition: all 0.4s ease-in-out; }
43 | .growl.growl-incoming {
44 | opacity: 0;
45 | filter: alpha(opacity=0); }
46 | .growl.growl-outgoing {
47 | opacity: 0;
48 | filter: alpha(opacity=0); }
49 | .growl.growl-small {
50 | width: 200px;
51 | padding: 5px;
52 | margin: 5px; }
53 | .growl.growl-medium {
54 | width: 250px;
55 | padding: 10px;
56 | margin: 10px; }
57 | .growl.growl-large {
58 | width: 300px;
59 | padding: 15px;
60 | margin: 15px; }
61 | .growl.growl-default {
62 | color: #FFF;
63 | background: #7f8c8d; }
64 | .growl.growl-error {
65 | color: #FFF;
66 | background: #C0392B; }
67 | .growl.growl-notice {
68 | color: #FFF;
69 | background: #2ECC71; }
70 | .growl.growl-warning {
71 | color: #FFF;
72 | background: #F39C12; }
73 | .growl .growl-close {
74 | cursor: pointer;
75 | float: right;
76 | font-size: 14px;
77 | line-height: 18px;
78 | font-weight: normal;
79 | font-family: helvetica, verdana, sans-serif; }
80 | .growl .growl-title {
81 | font-size: 18px;
82 | line-height: 24px; }
83 | .growl .growl-message {
84 | font-size: 14px;
85 | line-height: 18px; }
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/css/jumbotron-narrow.css:
--------------------------------------------------------------------------------
1 | /* Space out content a bit */
2 | body {
3 | padding-top: 20px;
4 | padding-bottom: 20px;
5 | }
6 |
7 | .avatar {
8 | width: 240px;
9 | height: 240px;
10 | }
11 |
12 | /* Everything but the jumbotron gets side spacing for mobile first views */
13 | .header,
14 | .marketing,
15 | .footer {
16 | padding-right: 15px;
17 | padding-left: 15px;
18 | }
19 |
20 | /* Custom page header */
21 | .header {
22 | padding-bottom: 20px;
23 | border-bottom: 1px solid #e5e5e5;
24 | }
25 | /* Make the masthead heading the same height as the navigation */
26 | .header h3 {
27 | margin-top: 0;
28 | margin-bottom: 0;
29 | line-height: 40px;
30 | }
31 |
32 | /* Custom page footer */
33 | .footer {
34 | padding-top: 19px;
35 | color: #777;
36 | border-top: 1px solid #e5e5e5;
37 | }
38 |
39 | /* Customize container */
40 | @media (min-width: 768px) {
41 | .container {
42 | max-width: 730px;
43 | }
44 | }
45 | .container-narrow > hr {
46 | margin: 30px 0;
47 | }
48 |
49 | /* Main marketing message and sign up button */
50 | .jumbotron {
51 | text-align: center;
52 | border-bottom: 1px solid #e5e5e5;
53 | }
54 | .jumbotron .btn {
55 | padding: 14px 24px;
56 | font-size: 21px;
57 | }
58 |
59 | /* Supporting marketing content */
60 | .marketing {
61 | margin: 40px 0;
62 | }
63 | .marketing p + h4 {
64 | margin-top: 28px;
65 | }
66 |
67 | /* Responsive: Portrait tablets and up */
68 | @media screen and (min-width: 768px) {
69 | /* Remove the padding we set earlier */
70 | .header,
71 | .marketing,
72 | .footer {
73 | padding-right: 0;
74 | padding-left: 0;
75 | }
76 | /* Space out the masthead */
77 | .header {
78 | margin-bottom: 30px;
79 | }
80 | /* Remove the bottom border on the jumbotron for visual effect */
81 | .jumbotron {
82 | border-bottom: 0;
83 | }
84 | }
85 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/css/password.css:
--------------------------------------------------------------------------------
1 | .jumbotron .btn {
2 | font-size: 21px;
3 | padding: 0px 0px;
4 | padding-top: 4px;
5 | }
6 |
7 | #password-btn {
8 | height: 34px;
9 | padding-right: 24px;
10 | padding-bottom: 18px;
11 | padding-left: 24px;
12 | font-size: 16px;
13 | }
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/css/signin.css:
--------------------------------------------------------------------------------
1 | body {
2 | padding-top: 40px;
3 | padding-bottom: 40px;
4 | background-color: #eee;
5 | }
6 |
7 | .form-signin {
8 | max-width: 330px;
9 | padding: 15px;
10 | margin: 0 auto;
11 | }
12 | .form-signin .form-signin-heading,
13 | .form-signin .checkbox {
14 | margin-bottom: 10px;
15 | }
16 | .form-signin .checkbox {
17 | font-weight: normal;
18 | }
19 | .form-signin .form-control {
20 | position: relative;
21 | height: auto;
22 | -webkit-box-sizing: border-box;
23 | -moz-box-sizing: border-box;
24 | box-sizing: border-box;
25 | padding: 10px;
26 | font-size: 16px;
27 | }
28 | .form-signin .form-control:focus {
29 | z-index: 2;
30 | }
31 | .form-signin input[type="email"] {
32 | margin-bottom: -1px;
33 | border-bottom-right-radius: 0;
34 | border-bottom-left-radius: 0;
35 | }
36 | .form-signin input[type="password"] {
37 | margin-bottom: 10px;
38 | border-top-left-radius: 0;
39 | border-top-right-radius: 0;
40 | }
41 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/auth0-samples/auth0-servlet-sample/932f880a72c65d7cc4937fd2c4d60d89f306c938/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.eot
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/auth0-samples/auth0-servlet-sample/932f880a72c65d7cc4937fd2c4d60d89f306c938/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.ttf
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/auth0-samples/auth0-servlet-sample/932f880a72c65d7cc4937fd2c4d60d89f306c938/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.woff
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/auth0-samples/auth0-servlet-sample/932f880a72c65d7cc4937fd2c4d60d89f306c938/01-Login/src/main/webapp/fonts/glyphicons-halflings-regular.woff2
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/js/bootstrap.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * Bootstrap v3.3.6 (http://getbootstrap.com)
3 | * Copyright 2011-2015 Twitter, Inc.
4 | * Licensed under the MIT license
5 | */
6 |
7 | if (typeof jQuery === 'undefined') {
8 | throw new Error('Bootstrap\'s JavaScript requires jQuery')
9 | }
10 |
11 | +function ($) {
12 | 'use strict';
13 | var version = $.fn.jquery.split(' ')[0].split('.')
14 | if ((version[0] < 2 && version[1] < 9) || (version[0] == 1 && version[1] == 9 && version[2] < 1) || (version[0] > 2)) {
15 | throw new Error('Bootstrap\'s JavaScript requires jQuery version 1.9.1 or higher, but lower than version 3')
16 | }
17 | }(jQuery);
18 |
19 | /* ========================================================================
20 | * Bootstrap: transition.js v3.3.6
21 | * http://getbootstrap.com/javascript/#transitions
22 | * ========================================================================
23 | * Copyright 2011-2015 Twitter, Inc.
24 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
25 | * ======================================================================== */
26 |
27 |
28 | +function ($) {
29 | 'use strict';
30 |
31 | // CSS TRANSITION SUPPORT (Shoutout: http://www.modernizr.com/)
32 | // ============================================================
33 |
34 | function transitionEnd() {
35 | var el = document.createElement('bootstrap')
36 |
37 | var transEndEventNames = {
38 | WebkitTransition : 'webkitTransitionEnd',
39 | MozTransition : 'transitionend',
40 | OTransition : 'oTransitionEnd otransitionend',
41 | transition : 'transitionend'
42 | }
43 |
44 | for (var name in transEndEventNames) {
45 | if (el.style[name] !== undefined) {
46 | return { end: transEndEventNames[name] }
47 | }
48 | }
49 |
50 | return false // explicit for ie8 ( ._.)
51 | }
52 |
53 | // http://blog.alexmaccaw.com/css-transitions
54 | $.fn.emulateTransitionEnd = function (duration) {
55 | var called = false
56 | var $el = this
57 | $(this).one('bsTransitionEnd', function () { called = true })
58 | var callback = function () { if (!called) $($el).trigger($.support.transition.end) }
59 | setTimeout(callback, duration)
60 | return this
61 | }
62 |
63 | $(function () {
64 | $.support.transition = transitionEnd()
65 |
66 | if (!$.support.transition) return
67 |
68 | $.event.special.bsTransitionEnd = {
69 | bindType: $.support.transition.end,
70 | delegateType: $.support.transition.end,
71 | handle: function (e) {
72 | if ($(e.target).is(this)) return e.handleObj.handler.apply(this, arguments)
73 | }
74 | }
75 | })
76 |
77 | }(jQuery);
78 |
79 | /* ========================================================================
80 | * Bootstrap: alert.js v3.3.6
81 | * http://getbootstrap.com/javascript/#alerts
82 | * ========================================================================
83 | * Copyright 2011-2015 Twitter, Inc.
84 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
85 | * ======================================================================== */
86 |
87 |
88 | +function ($) {
89 | 'use strict';
90 |
91 | // ALERT CLASS DEFINITION
92 | // ======================
93 |
94 | var dismiss = '[data-dismiss="alert"]'
95 | var Alert = function (el) {
96 | $(el).on('click', dismiss, this.close)
97 | }
98 |
99 | Alert.VERSION = '3.3.6'
100 |
101 | Alert.TRANSITION_DURATION = 150
102 |
103 | Alert.prototype.close = function (e) {
104 | var $this = $(this)
105 | var selector = $this.attr('data-target')
106 |
107 | if (!selector) {
108 | selector = $this.attr('href')
109 | selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') // strip for ie7
110 | }
111 |
112 | var $parent = $(selector)
113 |
114 | if (e) e.preventDefault()
115 |
116 | if (!$parent.length) {
117 | $parent = $this.closest('.alert')
118 | }
119 |
120 | $parent.trigger(e = $.Event('close.bs.alert'))
121 |
122 | if (e.isDefaultPrevented()) return
123 |
124 | $parent.removeClass('in')
125 |
126 | function removeElement() {
127 | // detach from parent, fire event then clean up data
128 | $parent.detach().trigger('closed.bs.alert').remove()
129 | }
130 |
131 | $.support.transition && $parent.hasClass('fade') ?
132 | $parent
133 | .one('bsTransitionEnd', removeElement)
134 | .emulateTransitionEnd(Alert.TRANSITION_DURATION) :
135 | removeElement()
136 | }
137 |
138 |
139 | // ALERT PLUGIN DEFINITION
140 | // =======================
141 |
142 | function Plugin(option) {
143 | return this.each(function () {
144 | var $this = $(this)
145 | var data = $this.data('bs.alert')
146 |
147 | if (!data) $this.data('bs.alert', (data = new Alert(this)))
148 | if (typeof option == 'string') data[option].call($this)
149 | })
150 | }
151 |
152 | var old = $.fn.alert
153 |
154 | $.fn.alert = Plugin
155 | $.fn.alert.Constructor = Alert
156 |
157 |
158 | // ALERT NO CONFLICT
159 | // =================
160 |
161 | $.fn.alert.noConflict = function () {
162 | $.fn.alert = old
163 | return this
164 | }
165 |
166 |
167 | // ALERT DATA-API
168 | // ==============
169 |
170 | $(document).on('click.bs.alert.data-api', dismiss, Alert.prototype.close)
171 |
172 | }(jQuery);
173 |
174 | /* ========================================================================
175 | * Bootstrap: button.js v3.3.6
176 | * http://getbootstrap.com/javascript/#buttons
177 | * ========================================================================
178 | * Copyright 2011-2015 Twitter, Inc.
179 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
180 | * ======================================================================== */
181 |
182 |
183 | +function ($) {
184 | 'use strict';
185 |
186 | // BUTTON PUBLIC CLASS DEFINITION
187 | // ==============================
188 |
189 | var Button = function (element, options) {
190 | this.$element = $(element)
191 | this.options = $.extend({}, Button.DEFAULTS, options)
192 | this.isLoading = false
193 | }
194 |
195 | Button.VERSION = '3.3.6'
196 |
197 | Button.DEFAULTS = {
198 | loadingText: 'loading...'
199 | }
200 |
201 | Button.prototype.setState = function (state) {
202 | var d = 'disabled'
203 | var $el = this.$element
204 | var val = $el.is('input') ? 'val' : 'html'
205 | var data = $el.data()
206 |
207 | state += 'Text'
208 |
209 | if (data.resetText == null) $el.data('resetText', $el[val]())
210 |
211 | // push to event loop to allow forms to submit
212 | setTimeout($.proxy(function () {
213 | $el[val](data[state] == null ? this.options[state] : data[state])
214 |
215 | if (state == 'loadingText') {
216 | this.isLoading = true
217 | $el.addClass(d).attr(d, d)
218 | } else if (this.isLoading) {
219 | this.isLoading = false
220 | $el.removeClass(d).removeAttr(d)
221 | }
222 | }, this), 0)
223 | }
224 |
225 | Button.prototype.toggle = function () {
226 | var changed = true
227 | var $parent = this.$element.closest('[data-toggle="buttons"]')
228 |
229 | if ($parent.length) {
230 | var $input = this.$element.find('input')
231 | if ($input.prop('type') == 'radio') {
232 | if ($input.prop('checked')) changed = false
233 | $parent.find('.active').removeClass('active')
234 | this.$element.addClass('active')
235 | } else if ($input.prop('type') == 'checkbox') {
236 | if (($input.prop('checked')) !== this.$element.hasClass('active')) changed = false
237 | this.$element.toggleClass('active')
238 | }
239 | $input.prop('checked', this.$element.hasClass('active'))
240 | if (changed) $input.trigger('change')
241 | } else {
242 | this.$element.attr('aria-pressed', !this.$element.hasClass('active'))
243 | this.$element.toggleClass('active')
244 | }
245 | }
246 |
247 |
248 | // BUTTON PLUGIN DEFINITION
249 | // ========================
250 |
251 | function Plugin(option) {
252 | return this.each(function () {
253 | var $this = $(this)
254 | var data = $this.data('bs.button')
255 | var options = typeof option == 'object' && option
256 |
257 | if (!data) $this.data('bs.button', (data = new Button(this, options)))
258 |
259 | if (option == 'toggle') data.toggle()
260 | else if (option) data.setState(option)
261 | })
262 | }
263 |
264 | var old = $.fn.button
265 |
266 | $.fn.button = Plugin
267 | $.fn.button.Constructor = Button
268 |
269 |
270 | // BUTTON NO CONFLICT
271 | // ==================
272 |
273 | $.fn.button.noConflict = function () {
274 | $.fn.button = old
275 | return this
276 | }
277 |
278 |
279 | // BUTTON DATA-API
280 | // ===============
281 |
282 | $(document)
283 | .on('click.bs.button.data-api', '[data-toggle^="button"]', function (e) {
284 | var $btn = $(e.target)
285 | if (!$btn.hasClass('btn')) $btn = $btn.closest('.btn')
286 | Plugin.call($btn, 'toggle')
287 | if (!($(e.target).is('input[type="radio"]') || $(e.target).is('input[type="checkbox"]'))) e.preventDefault()
288 | })
289 | .on('focus.bs.button.data-api blur.bs.button.data-api', '[data-toggle^="button"]', function (e) {
290 | $(e.target).closest('.btn').toggleClass('focus', /^focus(in)?$/.test(e.type))
291 | })
292 |
293 | }(jQuery);
294 |
295 | /* ========================================================================
296 | * Bootstrap: carousel.js v3.3.6
297 | * http://getbootstrap.com/javascript/#carousel
298 | * ========================================================================
299 | * Copyright 2011-2015 Twitter, Inc.
300 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
301 | * ======================================================================== */
302 |
303 |
304 | +function ($) {
305 | 'use strict';
306 |
307 | // CAROUSEL CLASS DEFINITION
308 | // =========================
309 |
310 | var Carousel = function (element, options) {
311 | this.$element = $(element)
312 | this.$indicators = this.$element.find('.carousel-indicators')
313 | this.options = options
314 | this.paused = null
315 | this.sliding = null
316 | this.interval = null
317 | this.$active = null
318 | this.$items = null
319 |
320 | this.options.keyboard && this.$element.on('keydown.bs.carousel', $.proxy(this.keydown, this))
321 |
322 | this.options.pause == 'hover' && !('ontouchstart' in document.documentElement) && this.$element
323 | .on('mouseenter.bs.carousel', $.proxy(this.pause, this))
324 | .on('mouseleave.bs.carousel', $.proxy(this.cycle, this))
325 | }
326 |
327 | Carousel.VERSION = '3.3.6'
328 |
329 | Carousel.TRANSITION_DURATION = 600
330 |
331 | Carousel.DEFAULTS = {
332 | interval: 5000,
333 | pause: 'hover',
334 | wrap: true,
335 | keyboard: true
336 | }
337 |
338 | Carousel.prototype.keydown = function (e) {
339 | if (/input|textarea/i.test(e.target.tagName)) return
340 | switch (e.which) {
341 | case 37: this.prev(); break
342 | case 39: this.next(); break
343 | default: return
344 | }
345 |
346 | e.preventDefault()
347 | }
348 |
349 | Carousel.prototype.cycle = function (e) {
350 | e || (this.paused = false)
351 |
352 | this.interval && clearInterval(this.interval)
353 |
354 | this.options.interval
355 | && !this.paused
356 | && (this.interval = setInterval($.proxy(this.next, this), this.options.interval))
357 |
358 | return this
359 | }
360 |
361 | Carousel.prototype.getItemIndex = function (item) {
362 | this.$items = item.parent().children('.item')
363 | return this.$items.index(item || this.$active)
364 | }
365 |
366 | Carousel.prototype.getItemForDirection = function (direction, active) {
367 | var activeIndex = this.getItemIndex(active)
368 | var willWrap = (direction == 'prev' && activeIndex === 0)
369 | || (direction == 'next' && activeIndex == (this.$items.length - 1))
370 | if (willWrap && !this.options.wrap) return active
371 | var delta = direction == 'prev' ? -1 : 1
372 | var itemIndex = (activeIndex + delta) % this.$items.length
373 | return this.$items.eq(itemIndex)
374 | }
375 |
376 | Carousel.prototype.to = function (pos) {
377 | var that = this
378 | var activeIndex = this.getItemIndex(this.$active = this.$element.find('.item.active'))
379 |
380 | if (pos > (this.$items.length - 1) || pos < 0) return
381 |
382 | if (this.sliding) return this.$element.one('slid.bs.carousel', function () { that.to(pos) }) // yes, "slid"
383 | if (activeIndex == pos) return this.pause().cycle()
384 |
385 | return this.slide(pos > activeIndex ? 'next' : 'prev', this.$items.eq(pos))
386 | }
387 |
388 | Carousel.prototype.pause = function (e) {
389 | e || (this.paused = true)
390 |
391 | if (this.$element.find('.next, .prev').length && $.support.transition) {
392 | this.$element.trigger($.support.transition.end)
393 | this.cycle(true)
394 | }
395 |
396 | this.interval = clearInterval(this.interval)
397 |
398 | return this
399 | }
400 |
401 | Carousel.prototype.next = function () {
402 | if (this.sliding) return
403 | return this.slide('next')
404 | }
405 |
406 | Carousel.prototype.prev = function () {
407 | if (this.sliding) return
408 | return this.slide('prev')
409 | }
410 |
411 | Carousel.prototype.slide = function (type, next) {
412 | var $active = this.$element.find('.item.active')
413 | var $next = next || this.getItemForDirection(type, $active)
414 | var isCycling = this.interval
415 | var direction = type == 'next' ? 'left' : 'right'
416 | var that = this
417 |
418 | if ($next.hasClass('active')) return (this.sliding = false)
419 |
420 | var relatedTarget = $next[0]
421 | var slideEvent = $.Event('slide.bs.carousel', {
422 | relatedTarget: relatedTarget,
423 | direction: direction
424 | })
425 | this.$element.trigger(slideEvent)
426 | if (slideEvent.isDefaultPrevented()) return
427 |
428 | this.sliding = true
429 |
430 | isCycling && this.pause()
431 |
432 | if (this.$indicators.length) {
433 | this.$indicators.find('.active').removeClass('active')
434 | var $nextIndicator = $(this.$indicators.children()[this.getItemIndex($next)])
435 | $nextIndicator && $nextIndicator.addClass('active')
436 | }
437 |
438 | var slidEvent = $.Event('slid.bs.carousel', { relatedTarget: relatedTarget, direction: direction }) // yes, "slid"
439 | if ($.support.transition && this.$element.hasClass('slide')) {
440 | $next.addClass(type)
441 | $next[0].offsetWidth // force reflow
442 | $active.addClass(direction)
443 | $next.addClass(direction)
444 | $active
445 | .one('bsTransitionEnd', function () {
446 | $next.removeClass([type, direction].join(' ')).addClass('active')
447 | $active.removeClass(['active', direction].join(' '))
448 | that.sliding = false
449 | setTimeout(function () {
450 | that.$element.trigger(slidEvent)
451 | }, 0)
452 | })
453 | .emulateTransitionEnd(Carousel.TRANSITION_DURATION)
454 | } else {
455 | $active.removeClass('active')
456 | $next.addClass('active')
457 | this.sliding = false
458 | this.$element.trigger(slidEvent)
459 | }
460 |
461 | isCycling && this.cycle()
462 |
463 | return this
464 | }
465 |
466 |
467 | // CAROUSEL PLUGIN DEFINITION
468 | // ==========================
469 |
470 | function Plugin(option) {
471 | return this.each(function () {
472 | var $this = $(this)
473 | var data = $this.data('bs.carousel')
474 | var options = $.extend({}, Carousel.DEFAULTS, $this.data(), typeof option == 'object' && option)
475 | var action = typeof option == 'string' ? option : options.slide
476 |
477 | if (!data) $this.data('bs.carousel', (data = new Carousel(this, options)))
478 | if (typeof option == 'number') data.to(option)
479 | else if (action) data[action]()
480 | else if (options.interval) data.pause().cycle()
481 | })
482 | }
483 |
484 | var old = $.fn.carousel
485 |
486 | $.fn.carousel = Plugin
487 | $.fn.carousel.Constructor = Carousel
488 |
489 |
490 | // CAROUSEL NO CONFLICT
491 | // ====================
492 |
493 | $.fn.carousel.noConflict = function () {
494 | $.fn.carousel = old
495 | return this
496 | }
497 |
498 |
499 | // CAROUSEL DATA-API
500 | // =================
501 |
502 | var clickHandler = function (e) {
503 | var href
504 | var $this = $(this)
505 | var $target = $($this.attr('data-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) // strip for ie7
506 | if (!$target.hasClass('carousel')) return
507 | var options = $.extend({}, $target.data(), $this.data())
508 | var slideIndex = $this.attr('data-slide-to')
509 | if (slideIndex) options.interval = false
510 |
511 | Plugin.call($target, options)
512 |
513 | if (slideIndex) {
514 | $target.data('bs.carousel').to(slideIndex)
515 | }
516 |
517 | e.preventDefault()
518 | }
519 |
520 | $(document)
521 | .on('click.bs.carousel.data-api', '[data-slide]', clickHandler)
522 | .on('click.bs.carousel.data-api', '[data-slide-to]', clickHandler)
523 |
524 | $(window).on('load', function () {
525 | $('[data-ride="carousel"]').each(function () {
526 | var $carousel = $(this)
527 | Plugin.call($carousel, $carousel.data())
528 | })
529 | })
530 |
531 | }(jQuery);
532 |
533 | /* ========================================================================
534 | * Bootstrap: collapse.js v3.3.6
535 | * http://getbootstrap.com/javascript/#collapse
536 | * ========================================================================
537 | * Copyright 2011-2015 Twitter, Inc.
538 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
539 | * ======================================================================== */
540 |
541 |
542 | +function ($) {
543 | 'use strict';
544 |
545 | // COLLAPSE PUBLIC CLASS DEFINITION
546 | // ================================
547 |
548 | var Collapse = function (element, options) {
549 | this.$element = $(element)
550 | this.options = $.extend({}, Collapse.DEFAULTS, options)
551 | this.$trigger = $('[data-toggle="collapse"][href="#' + element.id + '"],' +
552 | '[data-toggle="collapse"][data-target="#' + element.id + '"]')
553 | this.transitioning = null
554 |
555 | if (this.options.parent) {
556 | this.$parent = this.getParent()
557 | } else {
558 | this.addAriaAndCollapsedClass(this.$element, this.$trigger)
559 | }
560 |
561 | if (this.options.toggle) this.toggle()
562 | }
563 |
564 | Collapse.VERSION = '3.3.6'
565 |
566 | Collapse.TRANSITION_DURATION = 350
567 |
568 | Collapse.DEFAULTS = {
569 | toggle: true
570 | }
571 |
572 | Collapse.prototype.dimension = function () {
573 | var hasWidth = this.$element.hasClass('width')
574 | return hasWidth ? 'width' : 'height'
575 | }
576 |
577 | Collapse.prototype.show = function () {
578 | if (this.transitioning || this.$element.hasClass('in')) return
579 |
580 | var activesData
581 | var actives = this.$parent && this.$parent.children('.panel').children('.in, .collapsing')
582 |
583 | if (actives && actives.length) {
584 | activesData = actives.data('bs.collapse')
585 | if (activesData && activesData.transitioning) return
586 | }
587 |
588 | var startEvent = $.Event('show.bs.collapse')
589 | this.$element.trigger(startEvent)
590 | if (startEvent.isDefaultPrevented()) return
591 |
592 | if (actives && actives.length) {
593 | Plugin.call(actives, 'hide')
594 | activesData || actives.data('bs.collapse', null)
595 | }
596 |
597 | var dimension = this.dimension()
598 |
599 | this.$element
600 | .removeClass('collapse')
601 | .addClass('collapsing')[dimension](0)
602 | .attr('aria-expanded', true)
603 |
604 | this.$trigger
605 | .removeClass('collapsed')
606 | .attr('aria-expanded', true)
607 |
608 | this.transitioning = 1
609 |
610 | var complete = function () {
611 | this.$element
612 | .removeClass('collapsing')
613 | .addClass('collapse in')[dimension]('')
614 | this.transitioning = 0
615 | this.$element
616 | .trigger('shown.bs.collapse')
617 | }
618 |
619 | if (!$.support.transition) return complete.call(this)
620 |
621 | var scrollSize = $.camelCase(['scroll', dimension].join('-'))
622 |
623 | this.$element
624 | .one('bsTransitionEnd', $.proxy(complete, this))
625 | .emulateTransitionEnd(Collapse.TRANSITION_DURATION)[dimension](this.$element[0][scrollSize])
626 | }
627 |
628 | Collapse.prototype.hide = function () {
629 | if (this.transitioning || !this.$element.hasClass('in')) return
630 |
631 | var startEvent = $.Event('hide.bs.collapse')
632 | this.$element.trigger(startEvent)
633 | if (startEvent.isDefaultPrevented()) return
634 |
635 | var dimension = this.dimension()
636 |
637 | this.$element[dimension](this.$element[dimension]())[0].offsetHeight
638 |
639 | this.$element
640 | .addClass('collapsing')
641 | .removeClass('collapse in')
642 | .attr('aria-expanded', false)
643 |
644 | this.$trigger
645 | .addClass('collapsed')
646 | .attr('aria-expanded', false)
647 |
648 | this.transitioning = 1
649 |
650 | var complete = function () {
651 | this.transitioning = 0
652 | this.$element
653 | .removeClass('collapsing')
654 | .addClass('collapse')
655 | .trigger('hidden.bs.collapse')
656 | }
657 |
658 | if (!$.support.transition) return complete.call(this)
659 |
660 | this.$element
661 | [dimension](0)
662 | .one('bsTransitionEnd', $.proxy(complete, this))
663 | .emulateTransitionEnd(Collapse.TRANSITION_DURATION)
664 | }
665 |
666 | Collapse.prototype.toggle = function () {
667 | this[this.$element.hasClass('in') ? 'hide' : 'show']()
668 | }
669 |
670 | Collapse.prototype.getParent = function () {
671 | return $(this.options.parent)
672 | .find('[data-toggle="collapse"][data-parent="' + this.options.parent + '"]')
673 | .each($.proxy(function (i, element) {
674 | var $element = $(element)
675 | this.addAriaAndCollapsedClass(getTargetFromTrigger($element), $element)
676 | }, this))
677 | .end()
678 | }
679 |
680 | Collapse.prototype.addAriaAndCollapsedClass = function ($element, $trigger) {
681 | var isOpen = $element.hasClass('in')
682 |
683 | $element.attr('aria-expanded', isOpen)
684 | $trigger
685 | .toggleClass('collapsed', !isOpen)
686 | .attr('aria-expanded', isOpen)
687 | }
688 |
689 | function getTargetFromTrigger($trigger) {
690 | var href
691 | var target = $trigger.attr('data-target')
692 | || (href = $trigger.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') // strip for ie7
693 |
694 | return $(target)
695 | }
696 |
697 |
698 | // COLLAPSE PLUGIN DEFINITION
699 | // ==========================
700 |
701 | function Plugin(option) {
702 | return this.each(function () {
703 | var $this = $(this)
704 | var data = $this.data('bs.collapse')
705 | var options = $.extend({}, Collapse.DEFAULTS, $this.data(), typeof option == 'object' && option)
706 |
707 | if (!data && options.toggle && /show|hide/.test(option)) options.toggle = false
708 | if (!data) $this.data('bs.collapse', (data = new Collapse(this, options)))
709 | if (typeof option == 'string') data[option]()
710 | })
711 | }
712 |
713 | var old = $.fn.collapse
714 |
715 | $.fn.collapse = Plugin
716 | $.fn.collapse.Constructor = Collapse
717 |
718 |
719 | // COLLAPSE NO CONFLICT
720 | // ====================
721 |
722 | $.fn.collapse.noConflict = function () {
723 | $.fn.collapse = old
724 | return this
725 | }
726 |
727 |
728 | // COLLAPSE DATA-API
729 | // =================
730 |
731 | $(document).on('click.bs.collapse.data-api', '[data-toggle="collapse"]', function (e) {
732 | var $this = $(this)
733 |
734 | if (!$this.attr('data-target')) e.preventDefault()
735 |
736 | var $target = getTargetFromTrigger($this)
737 | var data = $target.data('bs.collapse')
738 | var option = data ? 'toggle' : $this.data()
739 |
740 | Plugin.call($target, option)
741 | })
742 |
743 | }(jQuery);
744 |
745 | /* ========================================================================
746 | * Bootstrap: dropdown.js v3.3.6
747 | * http://getbootstrap.com/javascript/#dropdowns
748 | * ========================================================================
749 | * Copyright 2011-2015 Twitter, Inc.
750 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
751 | * ======================================================================== */
752 |
753 |
754 | +function ($) {
755 | 'use strict';
756 |
757 | // DROPDOWN CLASS DEFINITION
758 | // =========================
759 |
760 | var backdrop = '.dropdown-backdrop'
761 | var toggle = '[data-toggle="dropdown"]'
762 | var Dropdown = function (element) {
763 | $(element).on('click.bs.dropdown', this.toggle)
764 | }
765 |
766 | Dropdown.VERSION = '3.3.6'
767 |
768 | function getParent($this) {
769 | var selector = $this.attr('data-target')
770 |
771 | if (!selector) {
772 | selector = $this.attr('href')
773 | selector = selector && /#[A-Za-z]/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') // strip for ie7
774 | }
775 |
776 | var $parent = selector && $(selector)
777 |
778 | return $parent && $parent.length ? $parent : $this.parent()
779 | }
780 |
781 | function clearMenus(e) {
782 | if (e && e.which === 3) return
783 | $(backdrop).remove()
784 | $(toggle).each(function () {
785 | var $this = $(this)
786 | var $parent = getParent($this)
787 | var relatedTarget = { relatedTarget: this }
788 |
789 | if (!$parent.hasClass('open')) return
790 |
791 | if (e && e.type == 'click' && /input|textarea/i.test(e.target.tagName) && $.contains($parent[0], e.target)) return
792 |
793 | $parent.trigger(e = $.Event('hide.bs.dropdown', relatedTarget))
794 |
795 | if (e.isDefaultPrevented()) return
796 |
797 | $this.attr('aria-expanded', 'false')
798 | $parent.removeClass('open').trigger($.Event('hidden.bs.dropdown', relatedTarget))
799 | })
800 | }
801 |
802 | Dropdown.prototype.toggle = function (e) {
803 | var $this = $(this)
804 |
805 | if ($this.is('.disabled, :disabled')) return
806 |
807 | var $parent = getParent($this)
808 | var isActive = $parent.hasClass('open')
809 |
810 | clearMenus()
811 |
812 | if (!isActive) {
813 | if ('ontouchstart' in document.documentElement && !$parent.closest('.navbar-nav').length) {
814 | // if mobile we use a backdrop because click events don't delegate
815 | $(document.createElement('div'))
816 | .addClass('dropdown-backdrop')
817 | .insertAfter($(this))
818 | .on('click', clearMenus)
819 | }
820 |
821 | var relatedTarget = { relatedTarget: this }
822 | $parent.trigger(e = $.Event('show.bs.dropdown', relatedTarget))
823 |
824 | if (e.isDefaultPrevented()) return
825 |
826 | $this
827 | .trigger('focus')
828 | .attr('aria-expanded', 'true')
829 |
830 | $parent
831 | .toggleClass('open')
832 | .trigger($.Event('shown.bs.dropdown', relatedTarget))
833 | }
834 |
835 | return false
836 | }
837 |
838 | Dropdown.prototype.keydown = function (e) {
839 | if (!/(38|40|27|32)/.test(e.which) || /input|textarea/i.test(e.target.tagName)) return
840 |
841 | var $this = $(this)
842 |
843 | e.preventDefault()
844 | e.stopPropagation()
845 |
846 | if ($this.is('.disabled, :disabled')) return
847 |
848 | var $parent = getParent($this)
849 | var isActive = $parent.hasClass('open')
850 |
851 | if (!isActive && e.which != 27 || isActive && e.which == 27) {
852 | if (e.which == 27) $parent.find(toggle).trigger('focus')
853 | return $this.trigger('click')
854 | }
855 |
856 | var desc = ' li:not(.disabled):visible a'
857 | var $items = $parent.find('.dropdown-menu' + desc)
858 |
859 | if (!$items.length) return
860 |
861 | var index = $items.index(e.target)
862 |
863 | if (e.which == 38 && index > 0) index-- // up
864 | if (e.which == 40 && index < $items.length - 1) index++ // down
865 | if (!~index) index = 0
866 |
867 | $items.eq(index).trigger('focus')
868 | }
869 |
870 |
871 | // DROPDOWN PLUGIN DEFINITION
872 | // ==========================
873 |
874 | function Plugin(option) {
875 | return this.each(function () {
876 | var $this = $(this)
877 | var data = $this.data('bs.dropdown')
878 |
879 | if (!data) $this.data('bs.dropdown', (data = new Dropdown(this)))
880 | if (typeof option == 'string') data[option].call($this)
881 | })
882 | }
883 |
884 | var old = $.fn.dropdown
885 |
886 | $.fn.dropdown = Plugin
887 | $.fn.dropdown.Constructor = Dropdown
888 |
889 |
890 | // DROPDOWN NO CONFLICT
891 | // ====================
892 |
893 | $.fn.dropdown.noConflict = function () {
894 | $.fn.dropdown = old
895 | return this
896 | }
897 |
898 |
899 | // APPLY TO STANDARD DROPDOWN ELEMENTS
900 | // ===================================
901 |
902 | $(document)
903 | .on('click.bs.dropdown.data-api', clearMenus)
904 | .on('click.bs.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
905 | .on('click.bs.dropdown.data-api', toggle, Dropdown.prototype.toggle)
906 | .on('keydown.bs.dropdown.data-api', toggle, Dropdown.prototype.keydown)
907 | .on('keydown.bs.dropdown.data-api', '.dropdown-menu', Dropdown.prototype.keydown)
908 |
909 | }(jQuery);
910 |
911 | /* ========================================================================
912 | * Bootstrap: modal.js v3.3.6
913 | * http://getbootstrap.com/javascript/#modals
914 | * ========================================================================
915 | * Copyright 2011-2015 Twitter, Inc.
916 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
917 | * ======================================================================== */
918 |
919 |
920 | +function ($) {
921 | 'use strict';
922 |
923 | // MODAL CLASS DEFINITION
924 | // ======================
925 |
926 | var Modal = function (element, options) {
927 | this.options = options
928 | this.$body = $(document.body)
929 | this.$element = $(element)
930 | this.$dialog = this.$element.find('.modal-dialog')
931 | this.$backdrop = null
932 | this.isShown = null
933 | this.originalBodyPad = null
934 | this.scrollbarWidth = 0
935 | this.ignoreBackdropClick = false
936 |
937 | if (this.options.remote) {
938 | this.$element
939 | .find('.modal-content')
940 | .load(this.options.remote, $.proxy(function () {
941 | this.$element.trigger('loaded.bs.modal')
942 | }, this))
943 | }
944 | }
945 |
946 | Modal.VERSION = '3.3.6'
947 |
948 | Modal.TRANSITION_DURATION = 300
949 | Modal.BACKDROP_TRANSITION_DURATION = 150
950 |
951 | Modal.DEFAULTS = {
952 | backdrop: true,
953 | keyboard: true,
954 | show: true
955 | }
956 |
957 | Modal.prototype.toggle = function (_relatedTarget) {
958 | return this.isShown ? this.hide() : this.show(_relatedTarget)
959 | }
960 |
961 | Modal.prototype.show = function (_relatedTarget) {
962 | var that = this
963 | var e = $.Event('show.bs.modal', { relatedTarget: _relatedTarget })
964 |
965 | this.$element.trigger(e)
966 |
967 | if (this.isShown || e.isDefaultPrevented()) return
968 |
969 | this.isShown = true
970 |
971 | this.checkScrollbar()
972 | this.setScrollbar()
973 | this.$body.addClass('modal-open')
974 |
975 | this.escape()
976 | this.resize()
977 |
978 | this.$element.on('click.dismiss.bs.modal', '[data-dismiss="modal"]', $.proxy(this.hide, this))
979 |
980 | this.$dialog.on('mousedown.dismiss.bs.modal', function () {
981 | that.$element.one('mouseup.dismiss.bs.modal', function (e) {
982 | if ($(e.target).is(that.$element)) that.ignoreBackdropClick = true
983 | })
984 | })
985 |
986 | this.backdrop(function () {
987 | var transition = $.support.transition && that.$element.hasClass('fade')
988 |
989 | if (!that.$element.parent().length) {
990 | that.$element.appendTo(that.$body) // don't move modals dom position
991 | }
992 |
993 | that.$element
994 | .show()
995 | .scrollTop(0)
996 |
997 | that.adjustDialog()
998 |
999 | if (transition) {
1000 | that.$element[0].offsetWidth // force reflow
1001 | }
1002 |
1003 | that.$element.addClass('in')
1004 |
1005 | that.enforceFocus()
1006 |
1007 | var e = $.Event('shown.bs.modal', { relatedTarget: _relatedTarget })
1008 |
1009 | transition ?
1010 | that.$dialog // wait for modal to slide in
1011 | .one('bsTransitionEnd', function () {
1012 | that.$element.trigger('focus').trigger(e)
1013 | })
1014 | .emulateTransitionEnd(Modal.TRANSITION_DURATION) :
1015 | that.$element.trigger('focus').trigger(e)
1016 | })
1017 | }
1018 |
1019 | Modal.prototype.hide = function (e) {
1020 | if (e) e.preventDefault()
1021 |
1022 | e = $.Event('hide.bs.modal')
1023 |
1024 | this.$element.trigger(e)
1025 |
1026 | if (!this.isShown || e.isDefaultPrevented()) return
1027 |
1028 | this.isShown = false
1029 |
1030 | this.escape()
1031 | this.resize()
1032 |
1033 | $(document).off('focusin.bs.modal')
1034 |
1035 | this.$element
1036 | .removeClass('in')
1037 | .off('click.dismiss.bs.modal')
1038 | .off('mouseup.dismiss.bs.modal')
1039 |
1040 | this.$dialog.off('mousedown.dismiss.bs.modal')
1041 |
1042 | $.support.transition && this.$element.hasClass('fade') ?
1043 | this.$element
1044 | .one('bsTransitionEnd', $.proxy(this.hideModal, this))
1045 | .emulateTransitionEnd(Modal.TRANSITION_DURATION) :
1046 | this.hideModal()
1047 | }
1048 |
1049 | Modal.prototype.enforceFocus = function () {
1050 | $(document)
1051 | .off('focusin.bs.modal') // guard against infinite focus loop
1052 | .on('focusin.bs.modal', $.proxy(function (e) {
1053 | if (this.$element[0] !== e.target && !this.$element.has(e.target).length) {
1054 | this.$element.trigger('focus')
1055 | }
1056 | }, this))
1057 | }
1058 |
1059 | Modal.prototype.escape = function () {
1060 | if (this.isShown && this.options.keyboard) {
1061 | this.$element.on('keydown.dismiss.bs.modal', $.proxy(function (e) {
1062 | e.which == 27 && this.hide()
1063 | }, this))
1064 | } else if (!this.isShown) {
1065 | this.$element.off('keydown.dismiss.bs.modal')
1066 | }
1067 | }
1068 |
1069 | Modal.prototype.resize = function () {
1070 | if (this.isShown) {
1071 | $(window).on('resize.bs.modal', $.proxy(this.handleUpdate, this))
1072 | } else {
1073 | $(window).off('resize.bs.modal')
1074 | }
1075 | }
1076 |
1077 | Modal.prototype.hideModal = function () {
1078 | var that = this
1079 | this.$element.hide()
1080 | this.backdrop(function () {
1081 | that.$body.removeClass('modal-open')
1082 | that.resetAdjustments()
1083 | that.resetScrollbar()
1084 | that.$element.trigger('hidden.bs.modal')
1085 | })
1086 | }
1087 |
1088 | Modal.prototype.removeBackdrop = function () {
1089 | this.$backdrop && this.$backdrop.remove()
1090 | this.$backdrop = null
1091 | }
1092 |
1093 | Modal.prototype.backdrop = function (callback) {
1094 | var that = this
1095 | var animate = this.$element.hasClass('fade') ? 'fade' : ''
1096 |
1097 | if (this.isShown && this.options.backdrop) {
1098 | var doAnimate = $.support.transition && animate
1099 |
1100 | this.$backdrop = $(document.createElement('div'))
1101 | .addClass('modal-backdrop ' + animate)
1102 | .appendTo(this.$body)
1103 |
1104 | this.$element.on('click.dismiss.bs.modal', $.proxy(function (e) {
1105 | if (this.ignoreBackdropClick) {
1106 | this.ignoreBackdropClick = false
1107 | return
1108 | }
1109 | if (e.target !== e.currentTarget) return
1110 | this.options.backdrop == 'static'
1111 | ? this.$element[0].focus()
1112 | : this.hide()
1113 | }, this))
1114 |
1115 | if (doAnimate) this.$backdrop[0].offsetWidth // force reflow
1116 |
1117 | this.$backdrop.addClass('in')
1118 |
1119 | if (!callback) return
1120 |
1121 | doAnimate ?
1122 | this.$backdrop
1123 | .one('bsTransitionEnd', callback)
1124 | .emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION) :
1125 | callback()
1126 |
1127 | } else if (!this.isShown && this.$backdrop) {
1128 | this.$backdrop.removeClass('in')
1129 |
1130 | var callbackRemove = function () {
1131 | that.removeBackdrop()
1132 | callback && callback()
1133 | }
1134 | $.support.transition && this.$element.hasClass('fade') ?
1135 | this.$backdrop
1136 | .one('bsTransitionEnd', callbackRemove)
1137 | .emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION) :
1138 | callbackRemove()
1139 |
1140 | } else if (callback) {
1141 | callback()
1142 | }
1143 | }
1144 |
1145 | // these following methods are used to handle overflowing modals
1146 |
1147 | Modal.prototype.handleUpdate = function () {
1148 | this.adjustDialog()
1149 | }
1150 |
1151 | Modal.prototype.adjustDialog = function () {
1152 | var modalIsOverflowing = this.$element[0].scrollHeight > document.documentElement.clientHeight
1153 |
1154 | this.$element.css({
1155 | paddingLeft: !this.bodyIsOverflowing && modalIsOverflowing ? this.scrollbarWidth : '',
1156 | paddingRight: this.bodyIsOverflowing && !modalIsOverflowing ? this.scrollbarWidth : ''
1157 | })
1158 | }
1159 |
1160 | Modal.prototype.resetAdjustments = function () {
1161 | this.$element.css({
1162 | paddingLeft: '',
1163 | paddingRight: ''
1164 | })
1165 | }
1166 |
1167 | Modal.prototype.checkScrollbar = function () {
1168 | var fullWindowWidth = window.innerWidth
1169 | if (!fullWindowWidth) { // workaround for missing window.innerWidth in IE8
1170 | var documentElementRect = document.documentElement.getBoundingClientRect()
1171 | fullWindowWidth = documentElementRect.right - Math.abs(documentElementRect.left)
1172 | }
1173 | this.bodyIsOverflowing = document.body.clientWidth < fullWindowWidth
1174 | this.scrollbarWidth = this.measureScrollbar()
1175 | }
1176 |
1177 | Modal.prototype.setScrollbar = function () {
1178 | var bodyPad = parseInt((this.$body.css('padding-right') || 0), 10)
1179 | this.originalBodyPad = document.body.style.paddingRight || ''
1180 | if (this.bodyIsOverflowing) this.$body.css('padding-right', bodyPad + this.scrollbarWidth)
1181 | }
1182 |
1183 | Modal.prototype.resetScrollbar = function () {
1184 | this.$body.css('padding-right', this.originalBodyPad)
1185 | }
1186 |
1187 | Modal.prototype.measureScrollbar = function () { // thx walsh
1188 | var scrollDiv = document.createElement('div')
1189 | scrollDiv.className = 'modal-scrollbar-measure'
1190 | this.$body.append(scrollDiv)
1191 | var scrollbarWidth = scrollDiv.offsetWidth - scrollDiv.clientWidth
1192 | this.$body[0].removeChild(scrollDiv)
1193 | return scrollbarWidth
1194 | }
1195 |
1196 |
1197 | // MODAL PLUGIN DEFINITION
1198 | // =======================
1199 |
1200 | function Plugin(option, _relatedTarget) {
1201 | return this.each(function () {
1202 | var $this = $(this)
1203 | var data = $this.data('bs.modal')
1204 | var options = $.extend({}, Modal.DEFAULTS, $this.data(), typeof option == 'object' && option)
1205 |
1206 | if (!data) $this.data('bs.modal', (data = new Modal(this, options)))
1207 | if (typeof option == 'string') data[option](_relatedTarget)
1208 | else if (options.show) data.show(_relatedTarget)
1209 | })
1210 | }
1211 |
1212 | var old = $.fn.modal
1213 |
1214 | $.fn.modal = Plugin
1215 | $.fn.modal.Constructor = Modal
1216 |
1217 |
1218 | // MODAL NO CONFLICT
1219 | // =================
1220 |
1221 | $.fn.modal.noConflict = function () {
1222 | $.fn.modal = old
1223 | return this
1224 | }
1225 |
1226 |
1227 | // MODAL DATA-API
1228 | // ==============
1229 |
1230 | $(document).on('click.bs.modal.data-api', '[data-toggle="modal"]', function (e) {
1231 | var $this = $(this)
1232 | var href = $this.attr('href')
1233 | var $target = $($this.attr('data-target') || (href && href.replace(/.*(?=#[^\s]+$)/, ''))) // strip for ie7
1234 | var option = $target.data('bs.modal') ? 'toggle' : $.extend({ remote: !/#/.test(href) && href }, $target.data(), $this.data())
1235 |
1236 | if ($this.is('a')) e.preventDefault()
1237 |
1238 | $target.one('show.bs.modal', function (showEvent) {
1239 | if (showEvent.isDefaultPrevented()) return // only register focus restorer if modal will actually get shown
1240 | $target.one('hidden.bs.modal', function () {
1241 | $this.is(':visible') && $this.trigger('focus')
1242 | })
1243 | })
1244 | Plugin.call($target, option, this)
1245 | })
1246 |
1247 | }(jQuery);
1248 |
1249 | /* ========================================================================
1250 | * Bootstrap: tooltip.js v3.3.6
1251 | * http://getbootstrap.com/javascript/#tooltip
1252 | * Inspired by the original jQuery.tipsy by Jason Frame
1253 | * ========================================================================
1254 | * Copyright 2011-2015 Twitter, Inc.
1255 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
1256 | * ======================================================================== */
1257 |
1258 |
1259 | +function ($) {
1260 | 'use strict';
1261 |
1262 | // TOOLTIP PUBLIC CLASS DEFINITION
1263 | // ===============================
1264 |
1265 | var Tooltip = function (element, options) {
1266 | this.type = null
1267 | this.options = null
1268 | this.enabled = null
1269 | this.timeout = null
1270 | this.hoverState = null
1271 | this.$element = null
1272 | this.inState = null
1273 |
1274 | this.init('tooltip', element, options)
1275 | }
1276 |
1277 | Tooltip.VERSION = '3.3.6'
1278 |
1279 | Tooltip.TRANSITION_DURATION = 150
1280 |
1281 | Tooltip.DEFAULTS = {
1282 | animation: true,
1283 | placement: 'top',
1284 | selector: false,
1285 | template: '',
1286 | trigger: 'hover focus',
1287 | title: '',
1288 | delay: 0,
1289 | html: false,
1290 | container: false,
1291 | viewport: {
1292 | selector: 'body',
1293 | padding: 0
1294 | }
1295 | }
1296 |
1297 | Tooltip.prototype.init = function (type, element, options) {
1298 | this.enabled = true
1299 | this.type = type
1300 | this.$element = $(element)
1301 | this.options = this.getOptions(options)
1302 | this.$viewport = this.options.viewport && $($.isFunction(this.options.viewport) ? this.options.viewport.call(this, this.$element) : (this.options.viewport.selector || this.options.viewport))
1303 | this.inState = { click: false, hover: false, focus: false }
1304 |
1305 | if (this.$element[0] instanceof document.constructor && !this.options.selector) {
1306 | throw new Error('`selector` option must be specified when initializing ' + this.type + ' on the window.document object!')
1307 | }
1308 |
1309 | var triggers = this.options.trigger.split(' ')
1310 |
1311 | for (var i = triggers.length; i--;) {
1312 | var trigger = triggers[i]
1313 |
1314 | if (trigger == 'click') {
1315 | this.$element.on('click.' + this.type, this.options.selector, $.proxy(this.toggle, this))
1316 | } else if (trigger != 'manual') {
1317 | var eventIn = trigger == 'hover' ? 'mouseenter' : 'focusin'
1318 | var eventOut = trigger == 'hover' ? 'mouseleave' : 'focusout'
1319 |
1320 | this.$element.on(eventIn + '.' + this.type, this.options.selector, $.proxy(this.enter, this))
1321 | this.$element.on(eventOut + '.' + this.type, this.options.selector, $.proxy(this.leave, this))
1322 | }
1323 | }
1324 |
1325 | this.options.selector ?
1326 | (this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
1327 | this.fixTitle()
1328 | }
1329 |
1330 | Tooltip.prototype.getDefaults = function () {
1331 | return Tooltip.DEFAULTS
1332 | }
1333 |
1334 | Tooltip.prototype.getOptions = function (options) {
1335 | options = $.extend({}, this.getDefaults(), this.$element.data(), options)
1336 |
1337 | if (options.delay && typeof options.delay == 'number') {
1338 | options.delay = {
1339 | show: options.delay,
1340 | hide: options.delay
1341 | }
1342 | }
1343 |
1344 | return options
1345 | }
1346 |
1347 | Tooltip.prototype.getDelegateOptions = function () {
1348 | var options = {}
1349 | var defaults = this.getDefaults()
1350 |
1351 | this._options && $.each(this._options, function (key, value) {
1352 | if (defaults[key] != value) options[key] = value
1353 | })
1354 |
1355 | return options
1356 | }
1357 |
1358 | Tooltip.prototype.enter = function (obj) {
1359 | var self = obj instanceof this.constructor ?
1360 | obj : $(obj.currentTarget).data('bs.' + this.type)
1361 |
1362 | if (!self) {
1363 | self = new this.constructor(obj.currentTarget, this.getDelegateOptions())
1364 | $(obj.currentTarget).data('bs.' + this.type, self)
1365 | }
1366 |
1367 | if (obj instanceof $.Event) {
1368 | self.inState[obj.type == 'focusin' ? 'focus' : 'hover'] = true
1369 | }
1370 |
1371 | if (self.tip().hasClass('in') || self.hoverState == 'in') {
1372 | self.hoverState = 'in'
1373 | return
1374 | }
1375 |
1376 | clearTimeout(self.timeout)
1377 |
1378 | self.hoverState = 'in'
1379 |
1380 | if (!self.options.delay || !self.options.delay.show) return self.show()
1381 |
1382 | self.timeout = setTimeout(function () {
1383 | if (self.hoverState == 'in') self.show()
1384 | }, self.options.delay.show)
1385 | }
1386 |
1387 | Tooltip.prototype.isInStateTrue = function () {
1388 | for (var key in this.inState) {
1389 | if (this.inState[key]) return true
1390 | }
1391 |
1392 | return false
1393 | }
1394 |
1395 | Tooltip.prototype.leave = function (obj) {
1396 | var self = obj instanceof this.constructor ?
1397 | obj : $(obj.currentTarget).data('bs.' + this.type)
1398 |
1399 | if (!self) {
1400 | self = new this.constructor(obj.currentTarget, this.getDelegateOptions())
1401 | $(obj.currentTarget).data('bs.' + this.type, self)
1402 | }
1403 |
1404 | if (obj instanceof $.Event) {
1405 | self.inState[obj.type == 'focusout' ? 'focus' : 'hover'] = false
1406 | }
1407 |
1408 | if (self.isInStateTrue()) return
1409 |
1410 | clearTimeout(self.timeout)
1411 |
1412 | self.hoverState = 'out'
1413 |
1414 | if (!self.options.delay || !self.options.delay.hide) return self.hide()
1415 |
1416 | self.timeout = setTimeout(function () {
1417 | if (self.hoverState == 'out') self.hide()
1418 | }, self.options.delay.hide)
1419 | }
1420 |
1421 | Tooltip.prototype.show = function () {
1422 | var e = $.Event('show.bs.' + this.type)
1423 |
1424 | if (this.hasContent() && this.enabled) {
1425 | this.$element.trigger(e)
1426 |
1427 | var inDom = $.contains(this.$element[0].ownerDocument.documentElement, this.$element[0])
1428 | if (e.isDefaultPrevented() || !inDom) return
1429 | var that = this
1430 |
1431 | var $tip = this.tip()
1432 |
1433 | var tipId = this.getUID(this.type)
1434 |
1435 | this.setContent()
1436 | $tip.attr('id', tipId)
1437 | this.$element.attr('aria-describedby', tipId)
1438 |
1439 | if (this.options.animation) $tip.addClass('fade')
1440 |
1441 | var placement = typeof this.options.placement == 'function' ?
1442 | this.options.placement.call(this, $tip[0], this.$element[0]) :
1443 | this.options.placement
1444 |
1445 | var autoToken = /\s?auto?\s?/i
1446 | var autoPlace = autoToken.test(placement)
1447 | if (autoPlace) placement = placement.replace(autoToken, '') || 'top'
1448 |
1449 | $tip
1450 | .detach()
1451 | .css({ top: 0, left: 0, display: 'block' })
1452 | .addClass(placement)
1453 | .data('bs.' + this.type, this)
1454 |
1455 | this.options.container ? $tip.appendTo(this.options.container) : $tip.insertAfter(this.$element)
1456 | this.$element.trigger('inserted.bs.' + this.type)
1457 |
1458 | var pos = this.getPosition()
1459 | var actualWidth = $tip[0].offsetWidth
1460 | var actualHeight = $tip[0].offsetHeight
1461 |
1462 | if (autoPlace) {
1463 | var orgPlacement = placement
1464 | var viewportDim = this.getPosition(this.$viewport)
1465 |
1466 | placement = placement == 'bottom' && pos.bottom + actualHeight > viewportDim.bottom ? 'top' :
1467 | placement == 'top' && pos.top - actualHeight < viewportDim.top ? 'bottom' :
1468 | placement == 'right' && pos.right + actualWidth > viewportDim.width ? 'left' :
1469 | placement == 'left' && pos.left - actualWidth < viewportDim.left ? 'right' :
1470 | placement
1471 |
1472 | $tip
1473 | .removeClass(orgPlacement)
1474 | .addClass(placement)
1475 | }
1476 |
1477 | var calculatedOffset = this.getCalculatedOffset(placement, pos, actualWidth, actualHeight)
1478 |
1479 | this.applyPlacement(calculatedOffset, placement)
1480 |
1481 | var complete = function () {
1482 | var prevHoverState = that.hoverState
1483 | that.$element.trigger('shown.bs.' + that.type)
1484 | that.hoverState = null
1485 |
1486 | if (prevHoverState == 'out') that.leave(that)
1487 | }
1488 |
1489 | $.support.transition && this.$tip.hasClass('fade') ?
1490 | $tip
1491 | .one('bsTransitionEnd', complete)
1492 | .emulateTransitionEnd(Tooltip.TRANSITION_DURATION) :
1493 | complete()
1494 | }
1495 | }
1496 |
1497 | Tooltip.prototype.applyPlacement = function (offset, placement) {
1498 | var $tip = this.tip()
1499 | var width = $tip[0].offsetWidth
1500 | var height = $tip[0].offsetHeight
1501 |
1502 | // manually read margins because getBoundingClientRect includes difference
1503 | var marginTop = parseInt($tip.css('margin-top'), 10)
1504 | var marginLeft = parseInt($tip.css('margin-left'), 10)
1505 |
1506 | // we must check for NaN for ie 8/9
1507 | if (isNaN(marginTop)) marginTop = 0
1508 | if (isNaN(marginLeft)) marginLeft = 0
1509 |
1510 | offset.top += marginTop
1511 | offset.left += marginLeft
1512 |
1513 | // $.fn.offset doesn't round pixel values
1514 | // so we use setOffset directly with our own function B-0
1515 | $.offset.setOffset($tip[0], $.extend({
1516 | using: function (props) {
1517 | $tip.css({
1518 | top: Math.round(props.top),
1519 | left: Math.round(props.left)
1520 | })
1521 | }
1522 | }, offset), 0)
1523 |
1524 | $tip.addClass('in')
1525 |
1526 | // check to see if placing tip in new offset caused the tip to resize itself
1527 | var actualWidth = $tip[0].offsetWidth
1528 | var actualHeight = $tip[0].offsetHeight
1529 |
1530 | if (placement == 'top' && actualHeight != height) {
1531 | offset.top = offset.top + height - actualHeight
1532 | }
1533 |
1534 | var delta = this.getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight)
1535 |
1536 | if (delta.left) offset.left += delta.left
1537 | else offset.top += delta.top
1538 |
1539 | var isVertical = /top|bottom/.test(placement)
1540 | var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight
1541 | var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'
1542 |
1543 | $tip.offset(offset)
1544 | this.replaceArrow(arrowDelta, $tip[0][arrowOffsetPosition], isVertical)
1545 | }
1546 |
1547 | Tooltip.prototype.replaceArrow = function (delta, dimension, isVertical) {
1548 | this.arrow()
1549 | .css(isVertical ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')
1550 | .css(isVertical ? 'top' : 'left', '')
1551 | }
1552 |
1553 | Tooltip.prototype.setContent = function () {
1554 | var $tip = this.tip()
1555 | var title = this.getTitle()
1556 |
1557 | $tip.find('.tooltip-inner')[this.options.html ? 'html' : 'text'](title)
1558 | $tip.removeClass('fade in top bottom left right')
1559 | }
1560 |
1561 | Tooltip.prototype.hide = function (callback) {
1562 | var that = this
1563 | var $tip = $(this.$tip)
1564 | var e = $.Event('hide.bs.' + this.type)
1565 |
1566 | function complete() {
1567 | if (that.hoverState != 'in') $tip.detach()
1568 | that.$element
1569 | .removeAttr('aria-describedby')
1570 | .trigger('hidden.bs.' + that.type)
1571 | callback && callback()
1572 | }
1573 |
1574 | this.$element.trigger(e)
1575 |
1576 | if (e.isDefaultPrevented()) return
1577 |
1578 | $tip.removeClass('in')
1579 |
1580 | $.support.transition && $tip.hasClass('fade') ?
1581 | $tip
1582 | .one('bsTransitionEnd', complete)
1583 | .emulateTransitionEnd(Tooltip.TRANSITION_DURATION) :
1584 | complete()
1585 |
1586 | this.hoverState = null
1587 |
1588 | return this
1589 | }
1590 |
1591 | Tooltip.prototype.fixTitle = function () {
1592 | var $e = this.$element
1593 | if ($e.attr('title') || typeof $e.attr('data-original-title') != 'string') {
1594 | $e.attr('data-original-title', $e.attr('title') || '').attr('title', '')
1595 | }
1596 | }
1597 |
1598 | Tooltip.prototype.hasContent = function () {
1599 | return this.getTitle()
1600 | }
1601 |
1602 | Tooltip.prototype.getPosition = function ($element) {
1603 | $element = $element || this.$element
1604 |
1605 | var el = $element[0]
1606 | var isBody = el.tagName == 'BODY'
1607 |
1608 | var elRect = el.getBoundingClientRect()
1609 | if (elRect.width == null) {
1610 | // width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093
1611 | elRect = $.extend({}, elRect, { width: elRect.right - elRect.left, height: elRect.bottom - elRect.top })
1612 | }
1613 | var elOffset = isBody ? { top: 0, left: 0 } : $element.offset()
1614 | var scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.scrollTop() }
1615 | var outerDims = isBody ? { width: $(window).width(), height: $(window).height() } : null
1616 |
1617 | return $.extend({}, elRect, scroll, outerDims, elOffset)
1618 | }
1619 |
1620 | Tooltip.prototype.getCalculatedOffset = function (placement, pos, actualWidth, actualHeight) {
1621 | return placement == 'bottom' ? { top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2 } :
1622 | placement == 'top' ? { top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2 } :
1623 | placement == 'left' ? { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth } :
1624 | /* placement == 'right' */ { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width }
1625 |
1626 | }
1627 |
1628 | Tooltip.prototype.getViewportAdjustedDelta = function (placement, pos, actualWidth, actualHeight) {
1629 | var delta = { top: 0, left: 0 }
1630 | if (!this.$viewport) return delta
1631 |
1632 | var viewportPadding = this.options.viewport && this.options.viewport.padding || 0
1633 | var viewportDimensions = this.getPosition(this.$viewport)
1634 |
1635 | if (/right|left/.test(placement)) {
1636 | var topEdgeOffset = pos.top - viewportPadding - viewportDimensions.scroll
1637 | var bottomEdgeOffset = pos.top + viewportPadding - viewportDimensions.scroll + actualHeight
1638 | if (topEdgeOffset < viewportDimensions.top) { // top overflow
1639 | delta.top = viewportDimensions.top - topEdgeOffset
1640 | } else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow
1641 | delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset
1642 | }
1643 | } else {
1644 | var leftEdgeOffset = pos.left - viewportPadding
1645 | var rightEdgeOffset = pos.left + viewportPadding + actualWidth
1646 | if (leftEdgeOffset < viewportDimensions.left) { // left overflow
1647 | delta.left = viewportDimensions.left - leftEdgeOffset
1648 | } else if (rightEdgeOffset > viewportDimensions.right) { // right overflow
1649 | delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset
1650 | }
1651 | }
1652 |
1653 | return delta
1654 | }
1655 |
1656 | Tooltip.prototype.getTitle = function () {
1657 | var title
1658 | var $e = this.$element
1659 | var o = this.options
1660 |
1661 | title = $e.attr('data-original-title')
1662 | || (typeof o.title == 'function' ? o.title.call($e[0]) : o.title)
1663 |
1664 | return title
1665 | }
1666 |
1667 | Tooltip.prototype.getUID = function (prefix) {
1668 | do prefix += ~~(Math.random() * 1000000)
1669 | while (document.getElementById(prefix))
1670 | return prefix
1671 | }
1672 |
1673 | Tooltip.prototype.tip = function () {
1674 | if (!this.$tip) {
1675 | this.$tip = $(this.options.template)
1676 | if (this.$tip.length != 1) {
1677 | throw new Error(this.type + ' `template` option must consist of exactly 1 top-level element!')
1678 | }
1679 | }
1680 | return this.$tip
1681 | }
1682 |
1683 | Tooltip.prototype.arrow = function () {
1684 | return (this.$arrow = this.$arrow || this.tip().find('.tooltip-arrow'))
1685 | }
1686 |
1687 | Tooltip.prototype.enable = function () {
1688 | this.enabled = true
1689 | }
1690 |
1691 | Tooltip.prototype.disable = function () {
1692 | this.enabled = false
1693 | }
1694 |
1695 | Tooltip.prototype.toggleEnabled = function () {
1696 | this.enabled = !this.enabled
1697 | }
1698 |
1699 | Tooltip.prototype.toggle = function (e) {
1700 | var self = this
1701 | if (e) {
1702 | self = $(e.currentTarget).data('bs.' + this.type)
1703 | if (!self) {
1704 | self = new this.constructor(e.currentTarget, this.getDelegateOptions())
1705 | $(e.currentTarget).data('bs.' + this.type, self)
1706 | }
1707 | }
1708 |
1709 | if (e) {
1710 | self.inState.click = !self.inState.click
1711 | if (self.isInStateTrue()) self.enter(self)
1712 | else self.leave(self)
1713 | } else {
1714 | self.tip().hasClass('in') ? self.leave(self) : self.enter(self)
1715 | }
1716 | }
1717 |
1718 | Tooltip.prototype.destroy = function () {
1719 | var that = this
1720 | clearTimeout(this.timeout)
1721 | this.hide(function () {
1722 | that.$element.off('.' + that.type).removeData('bs.' + that.type)
1723 | if (that.$tip) {
1724 | that.$tip.detach()
1725 | }
1726 | that.$tip = null
1727 | that.$arrow = null
1728 | that.$viewport = null
1729 | })
1730 | }
1731 |
1732 |
1733 | // TOOLTIP PLUGIN DEFINITION
1734 | // =========================
1735 |
1736 | function Plugin(option) {
1737 | return this.each(function () {
1738 | var $this = $(this)
1739 | var data = $this.data('bs.tooltip')
1740 | var options = typeof option == 'object' && option
1741 |
1742 | if (!data && /destroy|hide/.test(option)) return
1743 | if (!data) $this.data('bs.tooltip', (data = new Tooltip(this, options)))
1744 | if (typeof option == 'string') data[option]()
1745 | })
1746 | }
1747 |
1748 | var old = $.fn.tooltip
1749 |
1750 | $.fn.tooltip = Plugin
1751 | $.fn.tooltip.Constructor = Tooltip
1752 |
1753 |
1754 | // TOOLTIP NO CONFLICT
1755 | // ===================
1756 |
1757 | $.fn.tooltip.noConflict = function () {
1758 | $.fn.tooltip = old
1759 | return this
1760 | }
1761 |
1762 | }(jQuery);
1763 |
1764 | /* ========================================================================
1765 | * Bootstrap: popover.js v3.3.6
1766 | * http://getbootstrap.com/javascript/#popovers
1767 | * ========================================================================
1768 | * Copyright 2011-2015 Twitter, Inc.
1769 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
1770 | * ======================================================================== */
1771 |
1772 |
1773 | +function ($) {
1774 | 'use strict';
1775 |
1776 | // POPOVER PUBLIC CLASS DEFINITION
1777 | // ===============================
1778 |
1779 | var Popover = function (element, options) {
1780 | this.init('popover', element, options)
1781 | }
1782 |
1783 | if (!$.fn.tooltip) throw new Error('Popover requires tooltip.js')
1784 |
1785 | Popover.VERSION = '3.3.6'
1786 |
1787 | Popover.DEFAULTS = $.extend({}, $.fn.tooltip.Constructor.DEFAULTS, {
1788 | placement: 'right',
1789 | trigger: 'click',
1790 | content: '',
1791 | template: ''
1792 | })
1793 |
1794 |
1795 | // NOTE: POPOVER EXTENDS tooltip.js
1796 | // ================================
1797 |
1798 | Popover.prototype = $.extend({}, $.fn.tooltip.Constructor.prototype)
1799 |
1800 | Popover.prototype.constructor = Popover
1801 |
1802 | Popover.prototype.getDefaults = function () {
1803 | return Popover.DEFAULTS
1804 | }
1805 |
1806 | Popover.prototype.setContent = function () {
1807 | var $tip = this.tip()
1808 | var title = this.getTitle()
1809 | var content = this.getContent()
1810 |
1811 | $tip.find('.popover-title')[this.options.html ? 'html' : 'text'](title)
1812 | $tip.find('.popover-content').children().detach().end()[ // we use append for html objects to maintain js events
1813 | this.options.html ? (typeof content == 'string' ? 'html' : 'append') : 'text'
1814 | ](content)
1815 |
1816 | $tip.removeClass('fade top bottom left right in')
1817 |
1818 | // IE8 doesn't accept hiding via the `:empty` pseudo selector, we have to do
1819 | // this manually by checking the contents.
1820 | if (!$tip.find('.popover-title').html()) $tip.find('.popover-title').hide()
1821 | }
1822 |
1823 | Popover.prototype.hasContent = function () {
1824 | return this.getTitle() || this.getContent()
1825 | }
1826 |
1827 | Popover.prototype.getContent = function () {
1828 | var $e = this.$element
1829 | var o = this.options
1830 |
1831 | return $e.attr('data-content')
1832 | || (typeof o.content == 'function' ?
1833 | o.content.call($e[0]) :
1834 | o.content)
1835 | }
1836 |
1837 | Popover.prototype.arrow = function () {
1838 | return (this.$arrow = this.$arrow || this.tip().find('.arrow'))
1839 | }
1840 |
1841 |
1842 | // POPOVER PLUGIN DEFINITION
1843 | // =========================
1844 |
1845 | function Plugin(option) {
1846 | return this.each(function () {
1847 | var $this = $(this)
1848 | var data = $this.data('bs.popover')
1849 | var options = typeof option == 'object' && option
1850 |
1851 | if (!data && /destroy|hide/.test(option)) return
1852 | if (!data) $this.data('bs.popover', (data = new Popover(this, options)))
1853 | if (typeof option == 'string') data[option]()
1854 | })
1855 | }
1856 |
1857 | var old = $.fn.popover
1858 |
1859 | $.fn.popover = Plugin
1860 | $.fn.popover.Constructor = Popover
1861 |
1862 |
1863 | // POPOVER NO CONFLICT
1864 | // ===================
1865 |
1866 | $.fn.popover.noConflict = function () {
1867 | $.fn.popover = old
1868 | return this
1869 | }
1870 |
1871 | }(jQuery);
1872 |
1873 | /* ========================================================================
1874 | * Bootstrap: scrollspy.js v3.3.6
1875 | * http://getbootstrap.com/javascript/#scrollspy
1876 | * ========================================================================
1877 | * Copyright 2011-2015 Twitter, Inc.
1878 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
1879 | * ======================================================================== */
1880 |
1881 |
1882 | +function ($) {
1883 | 'use strict';
1884 |
1885 | // SCROLLSPY CLASS DEFINITION
1886 | // ==========================
1887 |
1888 | function ScrollSpy(element, options) {
1889 | this.$body = $(document.body)
1890 | this.$scrollElement = $(element).is(document.body) ? $(window) : $(element)
1891 | this.options = $.extend({}, ScrollSpy.DEFAULTS, options)
1892 | this.selector = (this.options.target || '') + ' .nav li > a'
1893 | this.offsets = []
1894 | this.targets = []
1895 | this.activeTarget = null
1896 | this.scrollHeight = 0
1897 |
1898 | this.$scrollElement.on('scroll.bs.scrollspy', $.proxy(this.process, this))
1899 | this.refresh()
1900 | this.process()
1901 | }
1902 |
1903 | ScrollSpy.VERSION = '3.3.6'
1904 |
1905 | ScrollSpy.DEFAULTS = {
1906 | offset: 10
1907 | }
1908 |
1909 | ScrollSpy.prototype.getScrollHeight = function () {
1910 | return this.$scrollElement[0].scrollHeight || Math.max(this.$body[0].scrollHeight, document.documentElement.scrollHeight)
1911 | }
1912 |
1913 | ScrollSpy.prototype.refresh = function () {
1914 | var that = this
1915 | var offsetMethod = 'offset'
1916 | var offsetBase = 0
1917 |
1918 | this.offsets = []
1919 | this.targets = []
1920 | this.scrollHeight = this.getScrollHeight()
1921 |
1922 | if (!$.isWindow(this.$scrollElement[0])) {
1923 | offsetMethod = 'position'
1924 | offsetBase = this.$scrollElement.scrollTop()
1925 | }
1926 |
1927 | this.$body
1928 | .find(this.selector)
1929 | .map(function () {
1930 | var $el = $(this)
1931 | var href = $el.data('target') || $el.attr('href')
1932 | var $href = /^#./.test(href) && $(href)
1933 |
1934 | return ($href
1935 | && $href.length
1936 | && $href.is(':visible')
1937 | && [[$href[offsetMethod]().top + offsetBase, href]]) || null
1938 | })
1939 | .sort(function (a, b) { return a[0] - b[0] })
1940 | .each(function () {
1941 | that.offsets.push(this[0])
1942 | that.targets.push(this[1])
1943 | })
1944 | }
1945 |
1946 | ScrollSpy.prototype.process = function () {
1947 | var scrollTop = this.$scrollElement.scrollTop() + this.options.offset
1948 | var scrollHeight = this.getScrollHeight()
1949 | var maxScroll = this.options.offset + scrollHeight - this.$scrollElement.height()
1950 | var offsets = this.offsets
1951 | var targets = this.targets
1952 | var activeTarget = this.activeTarget
1953 | var i
1954 |
1955 | if (this.scrollHeight != scrollHeight) {
1956 | this.refresh()
1957 | }
1958 |
1959 | if (scrollTop >= maxScroll) {
1960 | return activeTarget != (i = targets[targets.length - 1]) && this.activate(i)
1961 | }
1962 |
1963 | if (activeTarget && scrollTop < offsets[0]) {
1964 | this.activeTarget = null
1965 | return this.clear()
1966 | }
1967 |
1968 | for (i = offsets.length; i--;) {
1969 | activeTarget != targets[i]
1970 | && scrollTop >= offsets[i]
1971 | && (offsets[i + 1] === undefined || scrollTop < offsets[i + 1])
1972 | && this.activate(targets[i])
1973 | }
1974 | }
1975 |
1976 | ScrollSpy.prototype.activate = function (target) {
1977 | this.activeTarget = target
1978 |
1979 | this.clear()
1980 |
1981 | var selector = this.selector +
1982 | '[data-target="' + target + '"],' +
1983 | this.selector + '[href="' + target + '"]'
1984 |
1985 | var active = $(selector)
1986 | .parents('li')
1987 | .addClass('active')
1988 |
1989 | if (active.parent('.dropdown-menu').length) {
1990 | active = active
1991 | .closest('li.dropdown')
1992 | .addClass('active')
1993 | }
1994 |
1995 | active.trigger('activate.bs.scrollspy')
1996 | }
1997 |
1998 | ScrollSpy.prototype.clear = function () {
1999 | $(this.selector)
2000 | .parentsUntil(this.options.target, '.active')
2001 | .removeClass('active')
2002 | }
2003 |
2004 |
2005 | // SCROLLSPY PLUGIN DEFINITION
2006 | // ===========================
2007 |
2008 | function Plugin(option) {
2009 | return this.each(function () {
2010 | var $this = $(this)
2011 | var data = $this.data('bs.scrollspy')
2012 | var options = typeof option == 'object' && option
2013 |
2014 | if (!data) $this.data('bs.scrollspy', (data = new ScrollSpy(this, options)))
2015 | if (typeof option == 'string') data[option]()
2016 | })
2017 | }
2018 |
2019 | var old = $.fn.scrollspy
2020 |
2021 | $.fn.scrollspy = Plugin
2022 | $.fn.scrollspy.Constructor = ScrollSpy
2023 |
2024 |
2025 | // SCROLLSPY NO CONFLICT
2026 | // =====================
2027 |
2028 | $.fn.scrollspy.noConflict = function () {
2029 | $.fn.scrollspy = old
2030 | return this
2031 | }
2032 |
2033 |
2034 | // SCROLLSPY DATA-API
2035 | // ==================
2036 |
2037 | $(window).on('load.bs.scrollspy.data-api', function () {
2038 | $('[data-spy="scroll"]').each(function () {
2039 | var $spy = $(this)
2040 | Plugin.call($spy, $spy.data())
2041 | })
2042 | })
2043 |
2044 | }(jQuery);
2045 |
2046 | /* ========================================================================
2047 | * Bootstrap: tab.js v3.3.6
2048 | * http://getbootstrap.com/javascript/#tabs
2049 | * ========================================================================
2050 | * Copyright 2011-2015 Twitter, Inc.
2051 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
2052 | * ======================================================================== */
2053 |
2054 |
2055 | +function ($) {
2056 | 'use strict';
2057 |
2058 | // TAB CLASS DEFINITION
2059 | // ====================
2060 |
2061 | var Tab = function (element) {
2062 | // jscs:disable requireDollarBeforejQueryAssignment
2063 | this.element = $(element)
2064 | // jscs:enable requireDollarBeforejQueryAssignment
2065 | }
2066 |
2067 | Tab.VERSION = '3.3.6'
2068 |
2069 | Tab.TRANSITION_DURATION = 150
2070 |
2071 | Tab.prototype.show = function () {
2072 | var $this = this.element
2073 | var $ul = $this.closest('ul:not(.dropdown-menu)')
2074 | var selector = $this.data('target')
2075 |
2076 | if (!selector) {
2077 | selector = $this.attr('href')
2078 | selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') // strip for ie7
2079 | }
2080 |
2081 | if ($this.parent('li').hasClass('active')) return
2082 |
2083 | var $previous = $ul.find('.active:last a')
2084 | var hideEvent = $.Event('hide.bs.tab', {
2085 | relatedTarget: $this[0]
2086 | })
2087 | var showEvent = $.Event('show.bs.tab', {
2088 | relatedTarget: $previous[0]
2089 | })
2090 |
2091 | $previous.trigger(hideEvent)
2092 | $this.trigger(showEvent)
2093 |
2094 | if (showEvent.isDefaultPrevented() || hideEvent.isDefaultPrevented()) return
2095 |
2096 | var $target = $(selector)
2097 |
2098 | this.activate($this.closest('li'), $ul)
2099 | this.activate($target, $target.parent(), function () {
2100 | $previous.trigger({
2101 | type: 'hidden.bs.tab',
2102 | relatedTarget: $this[0]
2103 | })
2104 | $this.trigger({
2105 | type: 'shown.bs.tab',
2106 | relatedTarget: $previous[0]
2107 | })
2108 | })
2109 | }
2110 |
2111 | Tab.prototype.activate = function (element, container, callback) {
2112 | var $active = container.find('> .active')
2113 | var transition = callback
2114 | && $.support.transition
2115 | && ($active.length && $active.hasClass('fade') || !!container.find('> .fade').length)
2116 |
2117 | function next() {
2118 | $active
2119 | .removeClass('active')
2120 | .find('> .dropdown-menu > .active')
2121 | .removeClass('active')
2122 | .end()
2123 | .find('[data-toggle="tab"]')
2124 | .attr('aria-expanded', false)
2125 |
2126 | element
2127 | .addClass('active')
2128 | .find('[data-toggle="tab"]')
2129 | .attr('aria-expanded', true)
2130 |
2131 | if (transition) {
2132 | element[0].offsetWidth // reflow for transition
2133 | element.addClass('in')
2134 | } else {
2135 | element.removeClass('fade')
2136 | }
2137 |
2138 | if (element.parent('.dropdown-menu').length) {
2139 | element
2140 | .closest('li.dropdown')
2141 | .addClass('active')
2142 | .end()
2143 | .find('[data-toggle="tab"]')
2144 | .attr('aria-expanded', true)
2145 | }
2146 |
2147 | callback && callback()
2148 | }
2149 |
2150 | $active.length && transition ?
2151 | $active
2152 | .one('bsTransitionEnd', next)
2153 | .emulateTransitionEnd(Tab.TRANSITION_DURATION) :
2154 | next()
2155 |
2156 | $active.removeClass('in')
2157 | }
2158 |
2159 |
2160 | // TAB PLUGIN DEFINITION
2161 | // =====================
2162 |
2163 | function Plugin(option) {
2164 | return this.each(function () {
2165 | var $this = $(this)
2166 | var data = $this.data('bs.tab')
2167 |
2168 | if (!data) $this.data('bs.tab', (data = new Tab(this)))
2169 | if (typeof option == 'string') data[option]()
2170 | })
2171 | }
2172 |
2173 | var old = $.fn.tab
2174 |
2175 | $.fn.tab = Plugin
2176 | $.fn.tab.Constructor = Tab
2177 |
2178 |
2179 | // TAB NO CONFLICT
2180 | // ===============
2181 |
2182 | $.fn.tab.noConflict = function () {
2183 | $.fn.tab = old
2184 | return this
2185 | }
2186 |
2187 |
2188 | // TAB DATA-API
2189 | // ============
2190 |
2191 | var clickHandler = function (e) {
2192 | e.preventDefault()
2193 | Plugin.call($(this), 'show')
2194 | }
2195 |
2196 | $(document)
2197 | .on('click.bs.tab.data-api', '[data-toggle="tab"]', clickHandler)
2198 | .on('click.bs.tab.data-api', '[data-toggle="pill"]', clickHandler)
2199 |
2200 | }(jQuery);
2201 |
2202 | /* ========================================================================
2203 | * Bootstrap: affix.js v3.3.6
2204 | * http://getbootstrap.com/javascript/#affix
2205 | * ========================================================================
2206 | * Copyright 2011-2015 Twitter, Inc.
2207 | * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
2208 | * ======================================================================== */
2209 |
2210 |
2211 | +function ($) {
2212 | 'use strict';
2213 |
2214 | // AFFIX CLASS DEFINITION
2215 | // ======================
2216 |
2217 | var Affix = function (element, options) {
2218 | this.options = $.extend({}, Affix.DEFAULTS, options)
2219 |
2220 | this.$target = $(this.options.target)
2221 | .on('scroll.bs.affix.data-api', $.proxy(this.checkPosition, this))
2222 | .on('click.bs.affix.data-api', $.proxy(this.checkPositionWithEventLoop, this))
2223 |
2224 | this.$element = $(element)
2225 | this.affixed = null
2226 | this.unpin = null
2227 | this.pinnedOffset = null
2228 |
2229 | this.checkPosition()
2230 | }
2231 |
2232 | Affix.VERSION = '3.3.6'
2233 |
2234 | Affix.RESET = 'affix affix-top affix-bottom'
2235 |
2236 | Affix.DEFAULTS = {
2237 | offset: 0,
2238 | target: window
2239 | }
2240 |
2241 | Affix.prototype.getState = function (scrollHeight, height, offsetTop, offsetBottom) {
2242 | var scrollTop = this.$target.scrollTop()
2243 | var position = this.$element.offset()
2244 | var targetHeight = this.$target.height()
2245 |
2246 | if (offsetTop != null && this.affixed == 'top') return scrollTop < offsetTop ? 'top' : false
2247 |
2248 | if (this.affixed == 'bottom') {
2249 | if (offsetTop != null) return (scrollTop + this.unpin <= position.top) ? false : 'bottom'
2250 | return (scrollTop + targetHeight <= scrollHeight - offsetBottom) ? false : 'bottom'
2251 | }
2252 |
2253 | var initializing = this.affixed == null
2254 | var colliderTop = initializing ? scrollTop : position.top
2255 | var colliderHeight = initializing ? targetHeight : height
2256 |
2257 | if (offsetTop != null && scrollTop <= offsetTop) return 'top'
2258 | if (offsetBottom != null && (colliderTop + colliderHeight >= scrollHeight - offsetBottom)) return 'bottom'
2259 |
2260 | return false
2261 | }
2262 |
2263 | Affix.prototype.getPinnedOffset = function () {
2264 | if (this.pinnedOffset) return this.pinnedOffset
2265 | this.$element.removeClass(Affix.RESET).addClass('affix')
2266 | var scrollTop = this.$target.scrollTop()
2267 | var position = this.$element.offset()
2268 | return (this.pinnedOffset = position.top - scrollTop)
2269 | }
2270 |
2271 | Affix.prototype.checkPositionWithEventLoop = function () {
2272 | setTimeout($.proxy(this.checkPosition, this), 1)
2273 | }
2274 |
2275 | Affix.prototype.checkPosition = function () {
2276 | if (!this.$element.is(':visible')) return
2277 |
2278 | var height = this.$element.height()
2279 | var offset = this.options.offset
2280 | var offsetTop = offset.top
2281 | var offsetBottom = offset.bottom
2282 | var scrollHeight = Math.max($(document).height(), $(document.body).height())
2283 |
2284 | if (typeof offset != 'object') offsetBottom = offsetTop = offset
2285 | if (typeof offsetTop == 'function') offsetTop = offset.top(this.$element)
2286 | if (typeof offsetBottom == 'function') offsetBottom = offset.bottom(this.$element)
2287 |
2288 | var affix = this.getState(scrollHeight, height, offsetTop, offsetBottom)
2289 |
2290 | if (this.affixed != affix) {
2291 | if (this.unpin != null) this.$element.css('top', '')
2292 |
2293 | var affixType = 'affix' + (affix ? '-' + affix : '')
2294 | var e = $.Event(affixType + '.bs.affix')
2295 |
2296 | this.$element.trigger(e)
2297 |
2298 | if (e.isDefaultPrevented()) return
2299 |
2300 | this.affixed = affix
2301 | this.unpin = affix == 'bottom' ? this.getPinnedOffset() : null
2302 |
2303 | this.$element
2304 | .removeClass(Affix.RESET)
2305 | .addClass(affixType)
2306 | .trigger(affixType.replace('affix', 'affixed') + '.bs.affix')
2307 | }
2308 |
2309 | if (affix == 'bottom') {
2310 | this.$element.offset({
2311 | top: scrollHeight - height - offsetBottom
2312 | })
2313 | }
2314 | }
2315 |
2316 |
2317 | // AFFIX PLUGIN DEFINITION
2318 | // =======================
2319 |
2320 | function Plugin(option) {
2321 | return this.each(function () {
2322 | var $this = $(this)
2323 | var data = $this.data('bs.affix')
2324 | var options = typeof option == 'object' && option
2325 |
2326 | if (!data) $this.data('bs.affix', (data = new Affix(this, options)))
2327 | if (typeof option == 'string') data[option]()
2328 | })
2329 | }
2330 |
2331 | var old = $.fn.affix
2332 |
2333 | $.fn.affix = Plugin
2334 | $.fn.affix.Constructor = Affix
2335 |
2336 |
2337 | // AFFIX NO CONFLICT
2338 | // =================
2339 |
2340 | $.fn.affix.noConflict = function () {
2341 | $.fn.affix = old
2342 | return this
2343 | }
2344 |
2345 |
2346 | // AFFIX DATA-API
2347 | // ==============
2348 |
2349 | $(window).on('load', function () {
2350 | $('[data-spy="affix"]').each(function () {
2351 | var $spy = $(this)
2352 | var data = $spy.data()
2353 |
2354 | data.offset = data.offset || {}
2355 |
2356 | if (data.offsetBottom != null) data.offset.bottom = data.offsetBottom
2357 | if (data.offsetTop != null) data.offset.top = data.offsetTop
2358 |
2359 | Plugin.call($spy, data)
2360 | })
2361 | })
2362 |
2363 | }(jQuery);
2364 |
--------------------------------------------------------------------------------
/01-Login/src/main/webapp/js/jquery.growl.js:
--------------------------------------------------------------------------------
1 | // Generated by CoffeeScript 1.9.3
2 |
3 | /*
4 | jQuery Growl
5 | Copyright 2015 Kevin Sylvestre
6 | 1.3.1
7 | */
8 |
9 | (function() {
10 | "use strict";
11 | var $, Animation, Growl,
12 | bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
13 |
14 | $ = jQuery;
15 |
16 | Animation = (function() {
17 | function Animation() {}
18 |
19 | Animation.transitions = {
20 | "webkitTransition": "webkitTransitionEnd",
21 | "mozTransition": "mozTransitionEnd",
22 | "oTransition": "oTransitionEnd",
23 | "transition": "transitionend"
24 | };
25 |
26 | Animation.transition = function($el) {
27 | var el, ref, result, type;
28 | el = $el[0];
29 | ref = this.transitions;
30 | for (type in ref) {
31 | result = ref[type];
32 | if (el.style[type] != null) {
33 | return result;
34 | }
35 | }
36 | };
37 |
38 | return Animation;
39 |
40 | })();
41 |
42 | Growl = (function() {
43 | Growl.settings = {
44 | namespace: 'growl',
45 | duration: 3200,
46 | close: "×",
47 | location: "default",
48 | style: "default",
49 | size: "medium",
50 | delayOnHover: true
51 | };
52 |
53 | Growl.growl = function(settings) {
54 | if (settings == null) {
55 | settings = {};
56 | }
57 | this.initialize();
58 | return new Growl(settings);
59 | };
60 |
61 | Growl.initialize = function() {
62 | return $("body:not(:has(#growls))").append('
');
63 | };
64 |
65 | function Growl(settings) {
66 | if (settings == null) {
67 | settings = {};
68 | }
69 | this.container = bind(this.container, this);
70 | this.content = bind(this.content, this);
71 | this.html = bind(this.html, this);
72 | this.$growl = bind(this.$growl, this);
73 | this.$growls = bind(this.$growls, this);
74 | this.animate = bind(this.animate, this);
75 | this.remove = bind(this.remove, this);
76 | this.dismiss = bind(this.dismiss, this);
77 | this.present = bind(this.present, this);
78 | this.waitAndDismiss = bind(this.waitAndDismiss, this);
79 | this.cycle = bind(this.cycle, this);
80 | this.close = bind(this.close, this);
81 | this.click = bind(this.click, this);
82 | this.mouseLeave = bind(this.mouseLeave, this);
83 | this.mouseEnter = bind(this.mouseEnter, this);
84 | this.unbind = bind(this.unbind, this);
85 | this.bind = bind(this.bind, this);
86 | this.render = bind(this.render, this);
87 | this.settings = $.extend({}, Growl.settings, settings);
88 | this.$growls().attr('class', this.settings.location);
89 | this.render();
90 | }
91 |
92 | Growl.prototype.render = function() {
93 | var $growl;
94 | $growl = this.$growl();
95 | this.$growls().append($growl);
96 | if (this.settings.fixed) {
97 | this.present();
98 | } else {
99 | this.cycle();
100 | }
101 | };
102 |
103 | Growl.prototype.bind = function($growl) {
104 | if ($growl == null) {
105 | $growl = this.$growl();
106 | }
107 | $growl.on("click", this.click);
108 | if (this.settings.delayOnHover) {
109 | $growl.on("mouseenter", this.mouseEnter);
110 | $growl.on("mouseleave", this.mouseLeave);
111 | }
112 | return $growl.on("contextmenu", this.close).find("." + this.settings.namespace + "-close").on("click", this.close);
113 | };
114 |
115 | Growl.prototype.unbind = function($growl) {
116 | if ($growl == null) {
117 | $growl = this.$growl();
118 | }
119 | $growl.off("click", this.click);
120 | if (this.settings.delayOnHover) {
121 | $growl.off("mouseenter", this.mouseEnter);
122 | $growl.off("mouseleave", this.mouseLeave);
123 | }
124 | return $growl.off("contextmenu", this.close).find("." + this.settings.namespace + "-close").off("click", this.close);
125 | };
126 |
127 | Growl.prototype.mouseEnter = function(event) {
128 | var $growl;
129 | $growl = this.$growl();
130 | return $growl.stop(true, true);
131 | };
132 |
133 | Growl.prototype.mouseLeave = function(event) {
134 | return this.waitAndDismiss();
135 | };
136 |
137 | Growl.prototype.click = function(event) {
138 | if (this.settings.url != null) {
139 | event.preventDefault();
140 | event.stopPropagation();
141 | return window.open(this.settings.url);
142 | }
143 | };
144 |
145 | Growl.prototype.close = function(event) {
146 | var $growl;
147 | event.preventDefault();
148 | event.stopPropagation();
149 | $growl = this.$growl();
150 | return $growl.stop().queue(this.dismiss).queue(this.remove);
151 | };
152 |
153 | Growl.prototype.cycle = function() {
154 | var $growl;
155 | $growl = this.$growl();
156 | return $growl.queue(this.present).queue(this.waitAndDismiss());
157 | };
158 |
159 | Growl.prototype.waitAndDismiss = function() {
160 | var $growl;
161 | $growl = this.$growl();
162 | return $growl.delay(this.settings.duration).queue(this.dismiss).queue(this.remove);
163 | };
164 |
165 | Growl.prototype.present = function(callback) {
166 | var $growl;
167 | $growl = this.$growl();
168 | this.bind($growl);
169 | return this.animate($growl, this.settings.namespace + "-incoming", 'out', callback);
170 | };
171 |
172 | Growl.prototype.dismiss = function(callback) {
173 | var $growl;
174 | $growl = this.$growl();
175 | this.unbind($growl);
176 | return this.animate($growl, this.settings.namespace + "-outgoing", 'in', callback);
177 | };
178 |
179 | Growl.prototype.remove = function(callback) {
180 | this.$growl().remove();
181 | return callback();
182 | };
183 |
184 | Growl.prototype.animate = function($element, name, direction, callback) {
185 | var transition;
186 | if (direction == null) {
187 | direction = 'in';
188 | }
189 | transition = Animation.transition($element);
190 | $element[direction === 'in' ? 'removeClass' : 'addClass'](name);
191 | $element.offset().position;
192 | $element[direction === 'in' ? 'addClass' : 'removeClass'](name);
193 | if (callback == null) {
194 | return;
195 | }
196 | if (transition != null) {
197 | $element.one(transition, callback);
198 | } else {
199 | callback();
200 | }
201 | };
202 |
203 | Growl.prototype.$growls = function() {
204 | return this.$_growls != null ? this.$_growls : this.$_growls = $('#growls');
205 | };
206 |
207 | Growl.prototype.$growl = function() {
208 | return this.$_growl != null ? this.$_growl : this.$_growl = $(this.html());
209 | };
210 |
211 | Growl.prototype.html = function() {
212 | return this.container(this.content());
213 | };
214 |
215 | Growl.prototype.content = function() {
216 | return "" + this.settings.close + "
\n" + this.settings.title + "
\n" + this.settings.message + "
";
217 | };
218 |
219 | Growl.prototype.container = function(content) {
220 | return "\n " + content + "\n
";
221 | };
222 |
223 | return Growl;
224 |
225 | })();
226 |
227 | this.Growl = Growl;
228 |
229 | $.growl = function(options) {
230 | if (options == null) {
231 | options = {};
232 | }
233 | return Growl.growl(options);
234 | };
235 |
236 | $.growl.error = function(options) {
237 | var settings;
238 | if (options == null) {
239 | options = {};
240 | }
241 | settings = {
242 | title: "Error!",
243 | style: "error"
244 | };
245 | return $.growl($.extend(settings, options));
246 | };
247 |
248 | $.growl.notice = function(options) {
249 | var settings;
250 | if (options == null) {
251 | options = {};
252 | }
253 | settings = {
254 | title: "Notice!",
255 | style: "notice"
256 | };
257 | return $.growl($.extend(settings, options));
258 | };
259 |
260 | $.growl.warning = function(options) {
261 | var settings;
262 | if (options == null) {
263 | options = {};
264 | }
265 | settings = {
266 | title: "Warning!",
267 | style: "warning"
268 | };
269 | return $.growl($.extend(settings, options));
270 | };
271 |
272 | }).call(this);
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2016 Auth0 Samples
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Auth0 Servlet Samples
2 |
3 | [](https://circleci.com/gh/auth0-samples/auth0-servlet-sample)
4 |
5 | This sample demonstrates how to add authentication to a Java web app using Auth0.
6 |
7 | Check the [Java Quickstart](https://auth0.com/docs/quickstart/webapp/java-spring-mvc) to better understand of this sample.
8 |
9 | ## What is Auth0?
10 |
11 | Auth0 helps you to:
12 |
13 | * Add authentication with [multiple authentication sources](https://docs.auth0.com/identityproviders), either social like **Google, Facebook, Microsoft Account, LinkedIn, GitHub, Twitter, Box, Salesforce, amont others**, or enterprise identity systems like **Windows Azure AD, Google Apps, Active Directory, ADFS or any SAML Identity Provider**.
14 | * Add authentication through more traditional **[username/password databases](https://docs.auth0.com/mysql-connection-tutorial)**.
15 | * Add support for **[linking different user accounts](https://docs.auth0.com/link-accounts)** with the same user.
16 | * Support for generating signed [Json Web Tokens](https://docs.auth0.com/jwt) to call your APIs and **flow the user identity** securely.
17 | * Analytics of how, when and where users are logging in.
18 | * Pull data from other sources and add it to the user profile, through [JavaScript rules](https://docs.auth0.com/rules).
19 |
20 | ## Create a free account in Auth0
21 |
22 | 1. Go to [Auth0](https://auth0.com) and click Sign Up.
23 | 2. Use Google, GitHub or Microsoft Account to login.
24 |
25 | ## Issue Reporting
26 |
27 | If you have found a bug or if you have a feature request, please report them at this repository issues section. Please do not report security vulnerabilities on the public GitHub issue tracker. The [Responsible Disclosure Program](https://auth0.com/whitehat) details the procedure for disclosing security issues.
28 |
29 | ## Author
30 |
31 | [Auth0](https://auth0.com)
32 |
33 | ## License
34 |
35 | This project is licensed under the MIT license. See the [LICENSE](LICENSE.txt) file for more info.
36 |
--------------------------------------------------------------------------------