Skip to content

Commit f490270

Browse files
committed
Sharing Making cookie page work, adding support in v12 docs.
1 parent 890afa0 commit f490270

File tree

12 files changed

+395
-400
lines changed

12 files changed

+395
-400
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,292 @@
1+
/* global $ */
2+
3+
(() => {
4+
$(document).ready(function () {
5+
runAnalyticsAndBanner()
6+
})
7+
8+
function runAnalyticsAndBanner () {
9+
window.GPK_CONSENT_COOKIE_VERSION = 1
10+
11+
/* Name of the cookie to save users cookie preferences to. */
12+
const CONSENT_COOKIE_NAME = 'prototype_kit_docs_cookies_policy'
13+
14+
const TRACKING_LIVE_ID = 'UA-26179049-11'
15+
16+
const COOKIE_CATEGORIES = {
17+
analytics: ['_ga', '_gid', '_gat_UA-' + TRACKING_LIVE_ID],
18+
essential: [CONSENT_COOKIE_NAME]
19+
}
20+
const DEFAULT_COOKIE_CONSENT = {
21+
analytics: false
22+
}
23+
24+
function runAnalytics () {
25+
(function (w, d, s, l, i) {
26+
w[l] = w[l] || []
27+
w[l].push({
28+
'gtm.start': new Date().getTime(),
29+
event: 'gtm.js'
30+
})
31+
const f = d.getElementsByTagName(s)[0]
32+
const j = d.createElement(s)
33+
const dl = l !== 'dataLayer' ? '&l=' + l : ''
34+
j.async = true
35+
j.src =
36+
'https://www.googletagmanager.com/gtm.js?id=' + i + dl
37+
f.parentNode.insertBefore(j, f)
38+
})(window, document, 'script', 'dataLayer', 'GTM-TTGHPFQ')
39+
}
40+
41+
function Cookie (name, value, options) {
42+
if (typeof value !== 'undefined') {
43+
if (value === false || value === null) {
44+
deleteCookie(name)
45+
} else {
46+
// Default expiry date of 30 days
47+
if (typeof options === 'undefined') {
48+
options = { days: 30 }
49+
}
50+
setCookie(name, value, options)
51+
}
52+
} else {
53+
return getCookie(name)
54+
}
55+
}
56+
57+
function getConsentCookie () {
58+
const consentCookie = getCookie(CONSENT_COOKIE_NAME)
59+
let consentCookieObj
60+
61+
if (consentCookie) {
62+
try {
63+
consentCookieObj = JSON.parse(consentCookie)
64+
} catch (err) {
65+
return null
66+
}
67+
} else {
68+
return null
69+
}
70+
71+
return consentCookieObj
72+
}
73+
74+
function isValidConsentCookie (options) {
75+
return (options && options.version >= window.GPK_CONSENT_COOKIE_VERSION)
76+
}
77+
78+
/** Update the user's cookie preferences. */
79+
function setConsentCookie (options) {
80+
let cookieConsent = getConsentCookie()
81+
82+
if (!cookieConsent) {
83+
cookieConsent = JSON.parse(JSON.stringify(DEFAULT_COOKIE_CONSENT))
84+
}
85+
86+
// Merge current cookie preferences and new preferences
87+
for (const option in options) {
88+
cookieConsent[option] = options[option]
89+
}
90+
91+
// Essential cookies cannot be deselected, ignore this cookie type
92+
delete cookieConsent.essential
93+
94+
cookieConsent.version = window.GPK_CONSENT_COOKIE_VERSION
95+
96+
// Set the consent cookie
97+
setCookie(CONSENT_COOKIE_NAME, JSON.stringify(cookieConsent), { days: 365 })
98+
99+
// Update the other cookies
100+
resetCookies()
101+
}
102+
103+
/** Apply the user's cookie preferences
104+
*
105+
* Deletes any cookies the user has not consented to.
106+
*/
107+
function resetCookies () {
108+
let options = getConsentCookie()
109+
110+
// If no preferences or old version use the default
111+
if (!isValidConsentCookie(options)) {
112+
options = JSON.parse(JSON.stringify(DEFAULT_COOKIE_CONSENT))
113+
}
114+
115+
for (const cookieType in options) {
116+
if (cookieType === 'version') {
117+
continue
118+
}
119+
120+
// Essential cookies cannot be deselected, ignore this cookie type
121+
if (cookieType === 'essential') {
122+
continue
123+
}
124+
125+
// Initialise analytics if allowed
126+
if (cookieType === 'analytics' && options[cookieType]) {
127+
// Enable GA if allowed
128+
window['ga-disable-UA-' + TRACKING_LIVE_ID] = false
129+
runAnalytics()
130+
} else {
131+
// Disable GA if not allowed
132+
window['ga-disable-UA-' + TRACKING_LIVE_ID] = true
133+
}
134+
135+
if (!options[cookieType]) {
136+
// Fetch the cookies in that category
137+
const cookiesInCategory = COOKIE_CATEGORIES[cookieType]
138+
139+
cookiesInCategory.forEach(function (cookie) {
140+
// Delete cookie
141+
Cookie(cookie, null)
142+
})
143+
}
144+
}
145+
}
146+
147+
function userAllowsCookieCategory (cookieCategory, cookiePreferences) {
148+
// Essential cookies are always allowed
149+
if (cookieCategory === 'essential') {
150+
return true
151+
}
152+
153+
// Sometimes cookiePreferences is malformed in some of the tests, so we need to handle these
154+
try {
155+
return cookiePreferences[cookieCategory]
156+
} catch (e) {
157+
console.error(e)
158+
return false
159+
}
160+
}
161+
162+
function userAllowsCookie (cookieName) {
163+
// Always allow setting the consent cookie
164+
if (cookieName === CONSENT_COOKIE_NAME) {
165+
return true
166+
}
167+
168+
// Get the current cookie preferences
169+
let cookiePreferences = getConsentCookie()
170+
171+
// If no preferences or old version use the default
172+
if (!isValidConsentCookie(cookiePreferences)) {
173+
cookiePreferences = DEFAULT_COOKIE_CONSENT
174+
}
175+
176+
for (const category in COOKIE_CATEGORIES) {
177+
const cookiesInCategory = COOKIE_CATEGORIES[category]
178+
179+
if (cookiesInCategory.indexOf(cookieName) !== '-1') {
180+
return userAllowsCookieCategory(category, cookiePreferences)
181+
}
182+
}
183+
184+
// Deny the cookie if it is not known to us
185+
return false
186+
}
187+
188+
function getCookie (name) {
189+
const nameEQ = name + '='
190+
const cookies = document.cookie.split(';')
191+
for (let i = 0, len = cookies.length; i < len; i++) {
192+
let cookie = cookies[i]
193+
while (cookie.charAt(0) === ' ') {
194+
cookie = cookie.substring(1, cookie.length)
195+
}
196+
if (cookie.indexOf(nameEQ) === 0) {
197+
return decodeURIComponent(cookie.substring(nameEQ.length))
198+
}
199+
}
200+
return null
201+
}
202+
203+
function setCookie (name, value, options) {
204+
if (userAllowsCookie(name)) {
205+
if (typeof options === 'undefined') {
206+
options = {}
207+
}
208+
let cookieString = name + '=' + value + '; path=/'
209+
if (options.days) {
210+
const date = new Date()
211+
date.setTime(date.getTime() + (options.days * 24 * 60 * 60 * 1000))
212+
cookieString = cookieString + '; expires=' + date.toGMTString()
213+
}
214+
if (document.location.protocol === 'https:') {
215+
cookieString = cookieString + '; Secure'
216+
}
217+
document.cookie = cookieString
218+
}
219+
}
220+
221+
function deleteCookie (name) {
222+
if (Cookie(name)) {
223+
// Cookies need to be deleted in the same level of specificity in which they were set
224+
// If a cookie was set with a specified domain, it needs to be specified when deleted
225+
// If a cookie wasn't set with the domain attribute, it shouldn't be there when deleted
226+
// You can't tell if a cookie was set with a domain attribute or not, so try both options
227+
document.cookie = name + '=;expires=Thu, 01 Jan 1970 00:00:00 GMT;path=/'
228+
document.cookie = name + '=;expires=Thu, 01 Jan 1970 00:00:00 GMT;domain=' + window.location.hostname + ';path=/'
229+
document.cookie = name + '=;expires=Thu, 01 Jan 1970 00:00:00 GMT;domain=.' + window.location.hostname + ';path=/'
230+
}
231+
}
232+
233+
// Show the banner if there is no consent cookie or if it is outdated
234+
const currentConsentCookie = document.cookie.match(new RegExp('(^| )' + CONSENT_COOKIE_NAME + '=([^;]+)'))
235+
236+
const hasValidCookie = currentConsentCookie && isValidConsentCookie(JSON.parse(currentConsentCookie[2]))
237+
238+
resetCookies()
239+
240+
if (!currentConsentCookie || !hasValidCookie) {
241+
const cookieBanner = document.querySelector('.govuk-cookie-banner')
242+
if (cookieBanner && window.location.pathname !== '/docs/cookies') {
243+
cookieBanner.removeAttribute('hidden')
244+
} else {
245+
console.warn('No cookie banner found.')
246+
}
247+
}
248+
249+
const $cookieConsentForm = document.querySelector('.cookie-consent-update')
250+
const $cookieBanner = document.querySelector('.govuk-cookie-banner')
251+
const $acceptButton = $cookieBanner.querySelector('.cookie-banner-accept-button')
252+
const $rejectButton = $cookieBanner.querySelector('.cookie-banner-reject-button')
253+
const $cookieMessage = $cookieBanner.querySelector('.govuk-cookie-banner__message')
254+
const $cookieConfirmationAccept = $cookieBanner.querySelector('.js-cookie-banner-confirmation-accept')
255+
const $cookieConfirmationReject = $cookieBanner.querySelector('.js-cookie-banner-confirmation-reject')
256+
const $hideButtons = $cookieBanner.querySelectorAll('.js-cookie-banner-hide')
257+
258+
$acceptButton.addEventListener('click', (e) => {
259+
setConsentCookie({
260+
analytics: true
261+
})
262+
$cookieMessage.setAttribute('hidden', 'hidden')
263+
$cookieConfirmationAccept.removeAttribute('hidden')
264+
})
265+
$rejectButton.addEventListener('click', (e) => {
266+
setConsentCookie({
267+
analytics: false
268+
})
269+
$cookieMessage.setAttribute('hidden', 'hidden')
270+
$cookieConfirmationReject.removeAttribute('hidden')
271+
})
272+
$hideButtons.forEach($hideButton => $hideButton.addEventListener('click', (e) => {
273+
$cookieBanner.setAttribute('hidden', 'hidden')
274+
}))
275+
276+
if ($cookieConsentForm) {
277+
const $yesInput = $cookieConsentForm.querySelector('input[value="yes"]')
278+
const $noInput = $cookieConsentForm.querySelector('input[value="no"]')
279+
if (userAllowsCookie('analytics')) {
280+
$yesInput.checked = true
281+
} else {
282+
$noInput.checked = true
283+
}
284+
$cookieConsentForm.addEventListener('submit', (e) => {
285+
e.preventDefault()
286+
setConsentCookie({
287+
analytics: $yesInput.checked
288+
})
289+
})
290+
}
291+
}
292+
})()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,90 @@
1+
{% extends "layout.html" %}
2+
3+
{% block pageTitle %}
4+
Cookies – GOV.UK Prototype Kit
5+
{% endblock %}
6+
7+
{% block beforeContent %}
8+
{{ govukBreadcrumbs({
9+
items: [
10+
{
11+
text: "GOV.UK Prototype Kit",
12+
href: "./"
13+
}
14+
]
15+
}) }}
16+
{% endblock %}
17+
18+
{% block content %}
19+
20+
21+
<div class="govuk-grid-row">
22+
<div class="govuk-grid-column-two-thirds">
23+
24+
<h1 class="govuk-heading-xl">Cookies</h1>
25+
26+
<p>The GOV.UK Prototype Kit puts small files (known as ‘cookies’) on to your computer.</p>
27+
28+
<p>We use cookies to make prototypes work as well as possible.</p>
29+
30+
<p>These cookies aren’t used to identify you personally.</p>
31+
32+
<p><a href="https://ico.org.uk/for-the-public/online/cookies/">Find out how to manage cookies</a>.</p>
33+
34+
<h2 class="govuk-heading-l">Storing users' answers within prototypes</h2>
35+
36+
<p>To help prototypes work as well as possible, each prototype stores a cookie to remember what users select on
37+
question pages.</p>
38+
39+
<p>This allows prototypes to have branching pages and show users' answers on pages like the 'Check your answers'
40+
summary page.</p>
41+
42+
{{ govukTable({
43+
classes: "app-table--fixed",
44+
caption: "Cookies relating to storing users' answers within prototypes",
45+
captionClasses: "govuk-visually-hidden",
46+
head: [
47+
{ text: "Name" },
48+
{ text: "Purpose" },
49+
{ text: "Expires" }
50+
],
51+
rows: [
52+
[
53+
{ text: "govuk-prototype-kit-\<UNIQUE-ID\>" },
54+
{ text: "Storing users' answers from question pages" },
55+
{ text: "4 hours" }
56+
]
57+
]
58+
}) }}
59+
60+
<h2 class="govuk-heading-l">Change your cookie settings</h2>
61+
<form action="/form-handler" method="post" novalidate class="cookie-consent-update">
62+
63+
{{ govukRadios({
64+
name: "analytics-cookies",
65+
fieldset: {
66+
legend: {
67+
text: "Do you want to accept analytics cookies?",
68+
classes: "govuk-fieldset__legend--s"
69+
}
70+
},
71+
items: [
72+
{
73+
value: "yes",
74+
text: "Yes"
75+
},
76+
{
77+
value: "no",
78+
text: "No"
79+
}
80+
]
81+
}) }}
82+
83+
{{ govukButton({
84+
text: "Save cookie settings"
85+
}) }}
86+
</form>
87+
</div>
88+
</div>
89+
90+
{% endblock %}

0 commit comments

Comments
 (0)