From 0c54adfb07ff7c862162b918605a60f914ac51ee Mon Sep 17 00:00:00 2001 From: Berkeley Martinez Date: Tue, 12 Jan 2016 22:26:19 -0800 Subject: [PATCH] Add map aside to challenges --- client/less/main.less | 1 + client/less/map.less | 125 ++++++++++++++++++++ client/main.js | 9 ++ server/boot/challenge.js | 31 +++-- server/views/partials/challenge-footer.jade | 44 +++++++ server/views/partials/navbar.jade | 4 +- 6 files changed, 205 insertions(+), 9 deletions(-) create mode 100644 client/less/map.less diff --git a/client/less/main.less b/client/less/main.less index 12dc2e3c06..c01dc70836 100644 --- a/client/less/main.less +++ b/client/less/main.less @@ -1115,3 +1115,4 @@ code { @import "jobs.less"; @import "challenge.less"; @import "toastr.less"; +@import "map.less"; diff --git a/client/less/map.less b/client/less/map.less new file mode 100644 index 0000000000..92832b339c --- /dev/null +++ b/client/less/map.less @@ -0,0 +1,125 @@ +/* + * based off of https://github.com/gitterHQ/sidecar + * license: MIT + */ +.map-aside { + + z-index: 20000; + position: fixed; + top: 0; + left: 60%; + bottom: 0; + right: 0; + + display: flex; + flex-direction: row; + + background-color: @body-bg; + border-left: 1px solid #333; + box-shadow: -12px 0 18px 0 rgba(50, 50, 50, 0.3); + + transition: transform 0.3s cubic-bezier(0.16, 0.22, 0.22, 1.7); + + &.is-collapsed { + transform: translateX(110%); + } + + /* Add some "extension" so that there isn't a gap + * when we translate(via animation) more than 100% */ + &:after { + content: ''; + + z-index: -1; + position: absolute; + top: 0; + left: 100%; + bottom: 0; + right: -100%; + + background-color: @body-bg; + } + + @media (max-width: 1150px) { + left: 45%; + } + @media (max-width: 944px) { + left: 30%; + } + @media (max-width: 600px) { + left: 15%; + } + @media (max-width: 500px) { + left: 0; + border-left: none; + } + + & > .map-aside-container { + flex: 1; + width: 100%; + height: 100%; + + border: 0; + overflow: auto + } +} + +.map-aside-action-bar { + position: absolute; + top: 0; + left: 0; + right: 0; + + display: flex; + justify-content: flex-end; + + padding-bottom: 0.7em; + + background: linear-gradient(to bottom, #ffffff 0%, #ffffff 50%, rgba(255, 255, 255, 0) 100%); + + z-index: 100; +} + +.map-aside-action-item { + display: flex; + /* main axis */ + justify-content: center; + /* cross axis */ + align-items: center; + + width: 40px; + height: 40px; + + padding-left: 0; + padding-right: 0; + + opacity: 0.65; + background: none; + background-position: center center; + background-repeat: no-repeat; + background-size: 22px 22px; + border: 0; + outline: none; + + cursor: pointer; + cursor: hand; + + transition: all 0.2s ease; + + &:hover, + &:focus { + opacity: 1; + } + + &:active { + filter: hue-rotate(80deg) saturate(150); + } +} + +.map-aside-action-pop-out { + margin-right: -4px; + background-image: url() +} + +.map-aside-action-collapse { + background-image: url() +} diff --git a/client/main.js b/client/main.js index b677fc2b80..8ce8aedc49 100644 --- a/client/main.js +++ b/client/main.js @@ -259,4 +259,13 @@ $(document).ready(function() { window.ga('send', 'event', 'FB_LINK', 'SHARE', 'Facebook map share'); window.location.href = link; }); + + // map + $('#nav-map-btn').on('click', () => { + $('.map-aside').removeClass('is-collapsed'); + }); + + $('.map-aside-action-collapse').on('click', () => { + $('.map-aside').addClass('is-collapsed'); + }); }); diff --git a/server/boot/challenge.js b/server/boot/challenge.js index 34fc38f5b3..5422711956 100644 --- a/server/boot/challenge.js +++ b/server/boot/challenge.js @@ -189,6 +189,15 @@ function getRenderData$(user, challenge$, origChallengeName, solution) { }); } +function getCompletedChallengeIds(user = {}) { + // if user + // get the id's of all the users completed challenges + return !user.completedChallenges ? + [] : + _.uniq(user.completedChallenges) + .map(({ id, _id }) => id || _id); +} + // create a stream of an array of all the challenge blocks function getSuperBlocks$(challenge$, completedChallenges) { return challenge$ @@ -406,8 +415,19 @@ module.exports = function(app) { function showChallenge(req, res, next) { const solution = req.query.solution; + const completedChallenges = getCompletedChallengeIds(req.user); - getRenderData$(req.user, challenge$, req.params.challengeName, solution) + Observable.combineLatest( + getRenderData$(req.user, challenge$, req.params.challengeName, solution), + getSuperBlocks$(challenge$, completedChallenges), + ({ data, ...rest }, superBlocks) => ({ + ...rest, + data: { + ...data, + superBlocks + } + }) + ) .subscribe( ({ type, redirectUrl, message, data }) => { if (message) { @@ -553,14 +573,9 @@ module.exports = function(app) { ); } - function showMap({ user = {} }, res, next) { - // if user - // get the id's of all the users completed challenges - const completedChallenges = !user.completedChallenges ? - [] : - _.uniq(user.completedChallenges).map(({ id, _id }) => id || _id); + function showMap({ user }, res, next) { - getSuperBlocks$(challenge$, completedChallenges) + getSuperBlocks$(challenge$, getCompletedChallengeIds(user)) .subscribe( superBlocks => { res.render('map/show', { diff --git a/server/views/partials/challenge-footer.jade b/server/views/partials/challenge-footer.jade index ed6be20a3f..080b61a409 100644 --- a/server/views/partials/challenge-footer.jade +++ b/server/views/partials/challenge-footer.jade @@ -5,3 +5,47 @@ script. if (typeof localStorage !== 'undefined') { localStorage.setItem('currentDashedName', typeof common !== 'undefined' && common.dashedName || ''); } +aside.map-aside.is-collapsed + .map-aside-action-bar + a.map-aside-action-item.map-aside-action-pop-out(href='/map' target='_blank' aria-label='open map in new tap') + button.map-aside-action-item.map-aside-action-collapse(aria-label='close map aside') + .map-aside-container + .col-xs-10.col-xs-offset-1 + for superBlock, index in superBlocks + for challengeBlock in superBlock.blocks + h4.bold #{challengeBlock.name} (#{challengeBlock.time}) + for challenge in challengeBlock.challenges + if challenge.completed + p.text-primary.ion-checkmark-circled.padded-ionic-icon.negative-15(name="#{challenge.dashedName}")   + a(href="/challenges/#{challenge.dashedName}") + = challenge.title + span.sr-only= " Complete" + else + if challenge.type === "bonfire" + p.ion-asterisk.padded-ionic-icon.negative-15(name="#{challenge.dashedName}")   + a(name="#{challenge.dashedName}" href="/challenges/#{challenge.dashedName}" class=challenge.isComingSoon ? 'disabled' : '') + span= challenge.title + span.sr-only= " Incomplete" + if challenge.markNew + span.text-success.small     + strong + em New + if challengeBlock.isComingSoon + span.text-success.small     + strong + em Coming Soon + else + p.ion-ios-circle-outline.padded-ionic-icon.negative-15(name="#{challenge.dashedName}")   + a(name="#{challenge.dashedName}" href="/challenges/#{challenge.dashedName}" class=challenge.isComingSoon ? 'disabled' : '') + span= challenge.title + span.sr-only= " Incomplete" + if challenge.markNew + span.text-success.small     + strong + em New + if challengeBlock.isComingSoon + span.text-success.small     + strong + em Coming Soon + if (index < superBlocks.length) + hr diff --git a/server/views/partials/navbar.jade b/server/views/partials/navbar.jade index a248fcdf2f..ae73e565c2 100644 --- a/server/views/partials/navbar.jade +++ b/server/views/partials/navbar.jade @@ -9,7 +9,9 @@ nav.navbar.navbar-default.navbar-fixed-top.nav-height ul.nav.navbar-nav.navbar-right.hamburger-dropdown li a.learn-btn(href='#') Learn - li + li.hidden-xs + a#nav-map-btn(href='#' onclick='return false') Map + li.visible-xs a(href='/map') Map li.hidden-xs a#nav-chat-btn(href='#' onclick="return false") Chat