On this page, we'll add some more examples on how to customize the IdP Selection Page Template in the app's Page Template section.

Page templates are using Velocity (http://velocity.apache.org/) and support HTML, CSS, JS and images.
Images can only be embedded directly, as Base64 blobs. You can convert them here: https://www.base64-image.de/

Jira (IdP Buttons)

This is for Jira only, looks like the normal login page with IdPs added as Buttons

<html>
    <head>
        <title>Select Identity Provider</title>
        $webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
        <meta name="decorator" content="atl.general">
    </head>
    <body class="aui-layout aui-theme-default page-type-message">
        <section id="content" role="main">
            <div class="aui-page-panel">
                <div class="aui-page-panel-inner">
                    <section class="aui-page-panel-content">
                        <header class="aui-page-header">
                            <div class="aui-page-header-inner">
                                <div class="aui-page-header-main">
                                    <h1>Welcome to Jira</h1>
                                </div>
                            </div>
                        </header>
                        <form action="/login.jsp" class="aui" id="login-form" method="post">
                            <div class="form-body">
                                <div class="aui-group jira-login-method">
                                    <div class="aui-item jira-login-item">
                                        <div class="field-group">
                                            <label accesskey="u" for="login-form-username"><u>U</u>sername</label>
                                            <input class="text medium-field" id="login-form-username" name="os_username"
                                                   type="text" value="">
                                        </div>
                                        <div class="field-group">
                                            <label accesskey="p" for="login-form-password" id="passwordlabel"><u>P</u>assword</label>
                                            <input id="login-form-password" class="text medium-field" name="os_password"
                                                   type="password">
                                        </div>
                                        <fieldset class="group ">
                                            <div class="checkbox" resolved="">
                                                <input class="checkbox" id="login-form-remember-me" name="os_cookie"
                                                       type="checkbox" value="true" resolved=""><span
                                                    class="aui-form-glyph"></span>
                                                <label for="login-form-remember-me" accesskey="r"><u>R</u>emember my
                                                    login on this computer</label>
                                            </div>
                                        </fieldset>
                                        <!-- // .group -->
                                        <div id="sign-up-hint" class="field-group">
                                            Not a member? To request an account, please contact your Jira
                                            administrators.
                                        </div>
                                    </div>
                                </div>
                                <div class="hidden">
                                    <input name="os_destination" id="redirect_to" type="hidden" >
                                	<script> 
                                	    const queryString = window.location.search;
                                	    const urlParams = new URLSearchParams(queryString);
                                	    if (urlParams.has('redirectTo')){
                                	        document.getElementById('redirect_to').value = urlParams.get('redirectTo');
                                	    }
                                	</script>
                                </div>
                                <div class="hidden">
                                    <input name="user_role" type="hidden">
                                </div>
                                <div class="hidden">
                                    <input name="atl_token" type="hidden">
                                </div>
                            </div>
                            <div class="buttons-container form-footer">
                                <div class="buttons">
                                    <input accesskey="s" class="aui-button aui-button-primary" id="login-form-submit"
                                           name="login" title="Press Ctrl+Alt+s to submit this form" type="submit"
                                           value="Log In" resolved="">
                                    <a accesskey="`" class="aui-button aui-button-link cancel"
                                       href="/secure/ForgotLoginDetails.jspa" id="login-form-cancel"
                                       title="Press Ctrl+Alt+` to cancel" resolved="">Can't access your account?</a>
                                </div>
                            </div>
                        </form>
                </div>
                <div class="buttons-container form-footer">
                    <div class="buttons" align="center">
                        #foreach($idp in $idps)
                        <button title="Click to login with $idp.name" class="aui-button"
                                onclick="window.location.href = '$idp.ssoUrl';">Login with $idp.name
                        </button>
                        #end
                    </div>
                    <p> </p>
                </div>
                </form>
                <!-- // .aui #login-form -->
        </section>
        <!-- .aui-page-panel-content -->
        </div>
        <!-- .aui-page-panel-inner -->
        </div>
        <!-- .aui-page-panel -->
        </section>
        </div>
        </div>
        </section>
    </body>
</html>
XML


Template with embedded Images and CSS/ JS

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1">
    <title>Your page title</title>

    <script src="https://code.jquery.com/jquery-3.3.1.min.js" integrity="sha256-FgpCb/KJQlLNfOu91ta32o/NMZxltwRo8QtmkMRdAu8=" crossorigin="anonymous"></script>
    <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
    <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js" integrity="sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa" crossorigin="anonymous"></script>
    <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css">

    <style type="text/css">
        body {
    background-image: url('');
    background-repeat: repeat;
}

* {
    font-family: 'Blue Sky Standard', sans-serif;
}

.content-container {
    /*margin-top: 120px;*/
    top: 30%;
    position: absolute;
    background-color: transparent !important;
    color: #fff;
}

.heading {
    font-size: 25px;
    font-weight: lighter;
}

.select-box {
    margin-top: 10px;
    max-width: 650px;
    padding: 0;
    color: #fff;
    background: rgba(256, 256, 256, 0.2);
    border-radius: 4px;
    font-weight: lighter;
}

.select-box > .inner-select:first-child {
    border-bottom: 1px solid #000;
}

.inner-select {
    padding: 30px;
    transition: 0.5s all;
    color: #fff;
}

.select-type {
    margin-bottom: 0;
    font-size: 15px;
}

.click {
    font-size: 10px;
}

.tab {
    padding: 0;
}

.inner-select:hover {
    cursor: pointer;
    background: rgba(256, 256, 256, 0.9);
    transition: 0.5s all;
    color: black;
    font-weight: normal;
}

.desc-container {
    padding: 0;
    max-width: 500px;
}

.description {
    font-size: smaller;
    font-weight: lighter;
}

.logo {
    width: 75px;
}

.logo-container {
    padding: 20px;
}

.arrow {
    padding: 7px 0;
}

.click-arrow {
    font-size: 25px;
}
    </style>
</head>
<body>
    <div class="container-fluid">
        <div class="logo-container">
            <img src="" class="logo">
        </div>

        <div class="content-container col-sm-4 col-sm-offset-4">
            <h1 class="heading">Your heading</h1>
                 #if($idpSelected)
                  <p>Select or wait 3 seconds to use $selectedName <span class="aui-icon aui-icon-wait"></span></p>
                    <script>
                      var timeout = setTimeout("location.href = '$selectedUrl';", 3000);
                      window.onclick= function () { clearTimeout(timeout); } 
                    </script>
                  #end
            <div class="select-box col-xs-12">
                #foreach($idp in $idps)
                    <a class="inner-select col-xs-12" href="$idp.ssoUrl">
                        <div class="tab col-xs-11">
                            <span class="select-type">$idp.name</span><br/>
                            <span class="click">CLICK HERE</span>
                        </div>
                        <div class="arrow col-xs-1 pull-right">
                            <i class="fa fa-angle-right click-arrow"></i>
                        </div>
                    </a>
                #end
            </div>
            <div class="select-box col-xs-12">
                    <a class="inner-select col-xs-12" href="/login.jsp?nosso">
                        <div class="tab col-xs-11">
                            <span class="select-type">Login with local password</span><br/>
                            <span class="click">CLICK HERE</span>
                        </div>
                        <div class="arrow col-xs-1 pull-right">
                            <i class="fa fa-angle-right click-arrow"></i>
                        </div>
                    </a>
            </div>                   <div class="desc-container col-xs-12">
                <h1 class="heading">Status</h1>
                <span class="description">More text to be displayed here</span>
            </div>
        </div>
    </div>
</body>
</html>
XML

Change the decorator

Atlassian's applications support standard page decorators, allowing our plugin to generate new web pages with consistent decoration by the host application across the Atlassian products.

Default we are using the 'atl.general' decorator in our page templates. The general decorator can be used for the header and footer of general pages outside the administration UI.

atl.general decorator

general_decorator


If you want to remove the header and footer from the page, you need to change the decorator type. We will use the 'atl.popup' decorator for content that can be placed in a new browser popup window.

Just change the decorator type to hide the header and footer.

In the “IdP Selection Page Template”, replace the line:

<meta name="decorator" content="atl.general">
CODE

to

<meta name="decorator" content="atl.popup">
CODE

page_templates_popup

Look & feel

general decorator

decorator_general_lf

popup decorator

decorator_popup_lf

Click-thru Terms & Conditions Sign-up

Would it be possible to provide some kind of "Click-thru" agreement to a user which requires that they sign up to the terms of the site before getting general access?

For this scenario, we use the 'IdP Selection Page Template' (SAML Single Sign-On Plugin Configuration → Page Templates) and modify the default IdP Selection Page Template.

When a user now accesses Jira/Confluence for the first time, this page is displayed to him. By clicking "Okay", he will then be redirected to the first configured Identity Provider.

In doing so, we set a cookie that stores this. If the page is now visited a second time, it will be redirected directly to the Identity Provider as long as the cookie is still set.

So this means that each of your users will see this banner once (if the cookie is lost, the page is displayed again).


terms_of_service_idp_selection

Below you will find the template, please use it in the Page Templates (IdP Selection Page) section. You can easily customize the text or button text to your needs, as well as the color via CSS.

<html>
<head>
    <title>Terms of Service</title>
    $webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
    <meta name="decorator" content="atl.general">
</head>
<body class="aui-layout aui-theme-default page-type-message">

<style>
    .alert {
        padding: 20px;
        background-color: #0000ff;
        color: white;
    }


    .closebtn {
        margin-left: 15px;
        color: white;
        font-weight: bold;
        float: right;
        font-size: 22px;
        line-height: 20px;
        cursor: pointer;
        transition: 0.3s;
    }

    .closebtn:hover {
        color: black;
    }
</style>


<div class="alert" id="message">
    Your data will be processed for things. You will see this message only once. Click 'Okay' to acknowledge and proceed.<br></br> 
    <button onclick="function myFunction() {
        window.location.href = '$idps.get(0).ssoUrl'
    }
    myFunction()"
            type="button">
     Okay
    </button>
    #if($idpSelected)
    <script>

        document.getElementById("message").style.display = "none";
        window.location.href = '$idps.get(0).ssoUrl'
    </script>
    #end
</div>
</body>
</html>
XML

Multi-language IdP Selection Page

Would it be possible to have a multi-language version of the 'IdP Selection Page Template' (SAML Single Sign On Plugin Configuration → Page Templates)?

We currently don't have an option in the UI to support multi-language templates. However, you can use JavaScript inside the IdP Selection templates to achieve your goal.

SAML_Page_Template

Below you will find an example based on our default page template. Our example contains the languages German (de) and French (fr), however you can change or add the languages to your needs.

Multilanguage Template

<html>
 <head>
  <title>Select Identity Provider</title>
  $webResourceManager.requireResource("com.atlassian.auiplugin:aui-spinner")
  $webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
  <meta name="decorator" content="atl.general">
 </head>
 <body class="aui-layout aui-theme-default page-type-message" >
  <section id="content" role="main">
   <div class="aui-page-panel"><div class="aui-page-panel-inner">
    <section class="aui-page-panel-content">
     <div style="display: flex; justify-content: center;" id="initial-load-spinner">
        <aui-spinner size="large"></aui-spinner>
     </div>
     <div class="form-body" style="display: none;" id="form-body">
     <h1 data-translate="title">Select your Identity Provider</h1>
     #if($idpSelected)
      <p><span data-translate="wait">Select or wait 3 seconds to use $selectedName</span> <span class="aui-icon aui-icon-wait"></span></p>
        <script>
          var timeout = setTimeout("location.href = '$selectedUrl';", 3000);
          window.onclick= function () { clearTimeout(timeout); } 
        </script>
      #end
      #foreach($idp in $idps)
      <p>
        <a href="$idp.ssoUrl" data-translate="$idp.name">$idp.name</a> $idp.description
      </p>
      #end
      #if($loginurl)      <p>
       <a href="$loginurl" data-translate="nosso">Login with username and password</a>
      </p>
      #end
     </div>
   </section>
  </div>
 </div>
</section>
<script>
    AJS.toInit(function() {
        //                   TRANSLATION SETTINGS
        // The key to this template is that elements above whose text is supposed to be translated all have this data-translate attribute, where they get a translation key
        // The following map contains mappings from languages to translation keys to the actual translations.
        
        // Add your translations here. The keys ('de', 'fr') can be either locales ('de-DE', 'de-AT') or languages ('de', 'fr'). When in doubt, just use the language it's going to be used as a fallback
        var translations = {
            de: {
                title: "Wählen Sie Ihren Identity Provider",
                wait: "Wählen Sie einen Identity Provider aus oder warten Sie 3 Sekunden um $selectedName zu benutzen",
                nosso: "Login mit Benutzername und Passwort",
                // IdP names are allowed as well
                // If they contain non-alphanumeric characters, you'll have to write the name in "quotes"
                // the keys are case-sensitive!!!
                "Your IdP name": "Ihr IdP-Name"
            },
            fr: { // This translation has been generously provided by Google Translate, I would not trust it, but it's here as an example.
                title: "Choisissez votre fournisseur d'identité",
                wait: "Sélectionnez un fournisseur d'identité ou attendez 3 secondes pour utiliser $selectedName",
                nosso: "Connectez-vous avec nom d'utilisateur et mot de passe",
                "Your IdP name": "Votre nom IdP"
            }
        };
        // This is supposed to be the language present in the fields that are given above
        var defaultLanguage = "en";

        //                   TRANSLATION CODE
        // This is where the magic happens. Ideally, you shouldn't need to touch the code after this.
        
        // Detect user language with fallbacks for different levels of old browsers
        // This will be ordered from most preferred to least preferred
        var userLanguages = navigator.languages || [navigator.language] || [navigator.userLanguage] || [];
        
        // ensure default language can also match.
        translations[defaultLanguage] = {};
        
        var matchedLanguage = null;

        // Check if one the user's preferred languages is available
        userLanguages.forEach(function (languageString) {
            if (matchedLanguage === null && languageString in translations) {
                matchedLanguage = languageString
            }
        });
        
        // if no match was found, try converting the locale strings into language strings and see if that helps
        if (matchedLanguage === null) {
            userLanguages.forEach(function (languageString) {
                try {
                    var splitLanguageString = languageString.split('-')[0];
                    if (matchedLanguage === null && typeof splitLanguageString === 'string' && splitLanguageString in translations) {
                        matchedLanguage = splitLanguageString;
                    }
                } catch (e) {
                    console.warn("Error preparing user language");
                }
            });
        }
        
        // if any (non-default language) match was found, we set the translated strings in the UI
        if (matchedLanguage !== null && matchedLanguage !== defaultLanguage) {
            var translationsToUse = translations[matchedLanguage];
            AJS.$('[data-translate]').each(function() {
                var elem = AJS.$(this);
                var translationToUse = translationsToUse[elem.attr('data-translate')];
                if (!translationToUse) {
                    console.warn("Did not find a", matchedLanguage, "translation for", elem.attr('data-translate'));
                    return;
                }
                elem.text(translationToUse);
            })
        }
        // and now display the UI
        AJS.$("#initial-load-spinner").hide();
        AJS.$("#form-body").show();
    })
</script>
</body>
</html>
JS

A little more background / explanation

The script basically extends every element with text in it to have an attribute (data-translate) that contains a “translation key”, while the text is kept as the default language. After loading the page, it executes a JavaScript block. This contains the translations you want to have for each language and translation key (you can modify this according to the example and your need). Afterwards, there’s code that detects the user’s preferred language and does the substitution if necessary.


IdP Selection By Email Page with a Button

Applying the following template to the "IdP Selection By Email Page Template" would add a button that only upon clicking it, the redirection happens.

 

IdP Selection by Email with a Button

#disable_html_escaping()
<html>
  <head>
  <title>Enter your Email Address</title> 
  $webResourceManager.requireResource("$pluginproperties.pluginkey:resources")
  <meta name="decorator" content="atl.general">
	<script type="text/javascript">
	  var emailDomains = $emailDomainMap;
	  var ssoURL = "$ssourl";
	  
  	AJS.toInit(function () {
		// Hide these items
		jQuery('#notFound').hide();
		jQuery('#redirecting').hide();

		// The form is never submitted
		jQuery("#emailAddressForm").submit(function (e) {
			checkEmailField();
			e.preventDefault();
		});

	});

	function checkEmailField() {
		jQuery('#notFound').hide();
		jQuery('#redirecting').hide();

		// The input must have an @ and a . behind it to be ready to be checked
		var addressPattern = new RegExp(".*@.*\\...+");
		var enteredEmail = jQuery("#emailAddressField").val();
		if (!addressPattern.test(enteredEmail)) {
			return;
		}
		var atPosition = enteredEmail.indexOf('@');
		if (atPosition < 0) {
			return;
		}
		var emailDomain = enteredEmail.substring(atPosition + 1).toLowerCase();
		if (!emailDomain) {
			return;
		}

		var restEndpointUrl = emailDomains.__emailDomainCheck;
		jQuery.ajax({
			url: restEndpointUrl + emailDomain,
			dataType: "json",
			success: function (data) {
				console.log("Successfully retrieved IdP for given email domain", data);
				var idpId = data.idpId;
				if (idpId !== 0) {
					jQuery('#notFound').hide();
					jQuery('#redirecting').show();
					jQuery('#emailAddressField').prop('disabled', true);

					redirectUrl = ssoURL.replace('IDPID', idpId);
					// This avoids a warning message when starting the redirect
					window.onbeforeunload = function () {
					};
					window.location = redirectUrl;
				} else {
					jQuery('#notFound').show();
				}
			},
			error: function (xhr, errorText, err) {
				console.error("Error checking email domain: " + errorText, err);
				jQuery('#notFound').show();
			}
		});
	}

    </script>
	</head>
	<body class="aui-layout aui-theme-default page-type-message">
	  <section id="content" role="main"> 
	    <div class="aui-page-panel">
         <div class="aui-page-panel-inner">
           <section class="aui-page-panel-content">
             <h1>Single Sign On</h1>
             <form class="aui top-label" id="emailAddressForm" >
               <div class="field-group">
                 <label for="emailAddressField">Enter your email address</label>
                 <input type="text" class="text" id="emailAddressField" name="emailAddressField">
                 <input type="submit" class="aui-button aui-button-primary" value="Do SSO">
                 <div class="description">
                   <div id="notFound"   >No SSO destination found for this address</div>
                   <div id="redirecting"><span class="aui-icon aui-icon-wait"></span>Redirecting...</div>
                 </div>
               </div>
               #if($loginurl)
               <p><a href="$loginurl">Login with username and password</a></p>
               #end
             </form>
           </section> 
         </div>
       </div>
     </section>
	</body>
</html>
JS