- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
- 76
- 77
- 78
- 79
- 80
- 81
- 82
- 83
- 84
- 85
- 86
- 87
- 88
- 89
- 90
- 91
- 92
- 93
- 94
- 95
- 96
- 97
- 98
- 99
- 100
- 101
- 102
- 103
- 104
- 105
- 106
- 107
- 108
- 109
- 110
- 111
- 112
- 113
- 114
- 115
- 116
- 117
- 118
- 119
- 120
- 121
- 122
- 123
- 124
- 125
- 126
- 127
- 128
- 129
- 130
- 131
- 132
- 133
- 134
- 135
- 136
- 137
- 138
- 139
- 140
- 141
- 142
- 143
- 144
- 145
- 146
- 147
- 148
- 149
- 150
- 151
- 152
- 153
- 154
- 155
- 156
- 157
- 158
- 159
- 160
- 161
- 162
- 163
- 164
- 165
- 166
- 167
- 168
- 169
- 170
- 171
- 172
- 173
- 174
- 175
- 176
- 177
- 178
- 179
- 180
- 181
- 182
- 183
- 184
- 185
- 186
- 187
- 188
- 189
- 190
- 191
- 192
- 193
- 194
- 195
- 196
- 197
- 198
- 199
- 200
- 201
- 202
- 203
- 204
- 205
- 206
- 207
- 208
- 209
- 210
- 211
- 212
- 213
- 214
- 215
- 216
- 217
- 218
- 219
- 220
- 221
- 222
- 223
- 224
- 225
- 226
- 227
- 228
- 229
- 230
- 231
- 232
- 233
- 234
- 235
- 236
- 237
- 238
- 239
- 240
- 241
- 242
- 243
- 244
- 245
- 246
- 247
- 248
- 249
- 250
- 251
- 252
- 253
- 254
- 255
- 256
- 257
- 258
- 259
- 260
- 261
- 262
- 263
- 264
- 265
- 266
- 267
- 268
- 269
- 270
- 271
- 272
- 273
- 274
- 275
- 276
- 277
- 278
- 279
- 280
- 281
- 282
- 283
- 284
- 285
- 286
- 287
- 288
- 289
- 290
- 291
- 292
- 293
- 294
- 295
- 296
- 297
- 298
- 299
- 300
- 301
- 302
- 303
- 304
- 305
- 306
- 307
- 308
- 309
- 310
- 311
- 312
- 313
- 314
- 315
- 316
- 317
- 318
- 319
- 320
- 321
- 322
- 323
- 324
- 325
- 326
- 327
- 328
- 329
- 330
- 331
- 332
- 333
- 334
- 335
- 336
- 337
- 338
- 339
- 340
- 341
- 342
- 343
- 344
- 345
- 346
- 347
- 348
- 349
- 350
- 351
- 352
- 353
- 354
- 355
- 356
- 357
- 358
- 359
- 360
- 361
- 362
- 363
- 364
- 365
- 366
- 367
- 368
- 369
- 370
- 371
- 372
- 373
- 374
- 375
- 376
- 377
- 378
- 379
- 380
- 381
- 382
- 383
- 384
- 385
- 386
- 387
- 388
- 389
- 390
- 391
- 392
- 393
- 394
- 395
- 396
- 397
- 398
- 399
- 400
- 401
- 402
- 403
- 404
- 405
- 406
- 407
- 408
- 409
- 410
- 411
- 412
- 413
- 414
- 415
- 416
- 417
- 418
- 419
- 420
- 421
- 422
- 423
- 424
- 425
- 426
- 427
- 428
- 429
- 430
- 431
- 432
- 433
- 434
- 435
- 436
- 437
- 438
- 439
- 440
- 441
- 442
- 443
- 444
- 445
- 446
- 447
- 448
- 449
- 450
- 451
- 452
- 453
- 454
- 455
- 456
- 457
- 458
- 459
- 460
- 461
- 462
- 463
- 464
- 465
- 466
- 467
- 468
- 469
- 470
- 471
- 472
- 473
- 474
- 475
- 476
- 477
- 478
- 479
- 480
- 481
- 482
- 483
- 484
- 485
- 486
- 487
- 488
- 489
- 490
- 491
- 492
- 493
- 494
- 495
- 496
- 497
- 498
- 499
- 500
- 501
- 502
- 503
- 504
- 505
- 506
- 507
- 508
- 509
- 510
- 511
- 512
- 513
- 514
- 515
- 516
- 517
- 518
- 519
- 520
- 521
- 522
- 523
- 524
- 525
- 526
- 527
- 528
- 529
- 530
- 531
- 532
- 533
- 534
- 535
- 536
- 537
- 538
- 539
- 540
- 541
- 542
- 543
- 544
- 545
- 546
- 547
- 548
- 549
- 550
- 551
- 552
- 553
- 554
- 555
- 556
- 557
- 558
- 559
- 560
- 561
- 562
- 563
- 564
- 565
- 566
- 567
- 568
- 569
- 570
- 571
- 572
- 573
- 574
- 575
- 576
- 577
- 578
- 579
- 580
- 581
- 582
- 583
- 584
- 585
- 586
- 587
- 588
- 589
- 590
- 591
- 592
- 593
- 594
- 595
- 596
- 597
- 598
- 599
- 600
- 601
- 602
- 603
- 604
- 605
- 606
- 607
- 608
- 609
- 610
- 611
- 612
- 613
- 614
- 615
- 616
- 617
- 618
- 619
- 620
- 621
- 622
- 623
- 624
- 625
- 626
- 627
- 628
- 629
- 630
- 631
- 632
- 633
- 634
- 635
- 636
- 637
- 638
- 639
- 640
- 641
- 642
- 643
- 644
- 645
- 646
- 647
- 648
- 649
- 650
- 651
- 652
- 653
- 654
- 655
- 656
- 657
- 658
<template>
<div >
<!-- Only show masthead-offer if we have a message to show -->
<section class="masthead-offer" v-if="mastheadOfferMessage">
{{ mastheadOfferMessage }}
</section>
<!-- Masthead Component-->
<div class="masthead desktop"
v-if="!this.isMobile"
:style="{'background-image': `url(/data-models/us/masthead/desktop/${this.id}.jpg)`}">
</div>
<div class="masthead mobile"
v-else
:style="{'background-image': `url(/data-models/us/masthead/mobile/${this.id}.jpg)`}">
</div>
<!-- Holds lang select for desktop and mobile. and Desktop's languages list. -->
<section class="languages-wrap">
<div class="content">
<div>{{ locale.body['lang-select'] }}</div>
<div>
<SelectLanguage :langcode='langcode' />
</div>
</div>
</section>
<section class="products-container">
<div class="content">
<div class="products" >
<!-- <LoadingCircle v-if='!productList' /> -->
<!-- Loop through our products | ProductBox Component -->
<!-- We pass the whole products arra in case we need to mess with skus, ie, bonus month -->
<ProductBox v-for="(product, key) in products"
:msrp="product.msrp"
:price="product.price"
:level="product.lvl"
:name="productName && productName[product.lvl] || product.name"
:cartLink="product.cart"
:showFullPrice="fullPrice"
:bonusMonths="bonusMonth"
:xpay="parseInt($route.query.xpay) || xpay"
:email="$route.query.email"
:ribbonText="ribbonText && ribbonText[product.lvl]"
:offerText="offerText && offerText[product.lvl]"
:products="productList"
:iframeCart=true
:productFilter="productFilter"
:key="key"
/>
</div>
<p class="disclaimer">
{{ locale.body['product-disclaimer'] }}
</p>
<!-- Guarantee Component -->
<div class="guarantee-wrapper">
<Guarantee />
</div>
</div>
</section>
<section class="highlights">
<div class="content">
<h3>{{ locale.body.features.h2 }}</h3>
<div>
<div class="product-img" data-bg='/nuxt-assets/img/online-lockup.png'></div>
<div class="bullets">
<div>
<p>
{{ locale.body.features.p }}
</p>
<ul>
<li v-for='(li, i) in locale.body.features.bullets' :key=i v-html='li'></li>
</ul>
</div>
</div>
</div>
</div>
</section>
<section class="trusted">
<div class="content">
<h3>{{ locale.body.trusted.h2 }}</h3>
<div class="logos">
<img alt="tripadvisor" data-src="/nuxt-assets/img/logos/tripadvisor-2.png">
<img alt="fender" data-src="/nuxt-assets/img/logos/fender.svg">
<img alt="NASA" data-src="/nuxt-assets/img/logos/nasa.svg">
<img alt="calvinklein" data-src="/nuxt-assets/img/logos/ck.svg">
</div>
</div>
</section>
<!-- Company Quotes -->
<div class="company-quotes">
<div class="content">
<CompanyQuotes />
</div>
</div>
<LandingPageFooter />
<CartPanel />
</div>
</template>
<script>
// Import our Components
import SelectLanguage from '~/components/SelectLanguage.vue'
import ProductBox from '~/components/ProductBox.vue'
import CompanyQuotes from '~/components/quotes/CompanyQuotes.vue'
import Guarantee from '~/components/Guarantee.vue'
import LoadingCircle from '~/components/loading/LoadingCircle.vue'
import LandingPageFooter from '~/components/footers/LandingPageFooter.vue'
import CartPanel from '~/components/CartPanel.vue'
import locale from '~/data/airtable/landing-pages/sbsr/index.yaml'
export default {
layout: 'landing-page',
/* Components this page uses. They are imported above */
components: {
SelectLanguage,
ProductBox,
CompanyQuotes,
Guarantee,
CompanyQuotes,
LoadingCircle,
LandingPageFooter,
CartPanel
},
/**
* Sets head information for this page.
* Metadata, scripts, css, title, etc, can
* all be set and fetched here
*/
head () {
return {
title: 'Rosetta Stone - Learn a New Language',
meta: [
{ charset: 'utf-8' },
{ name: 'viewport', content: 'width=device-width, initial-scale=1' },
{ hid: 'description', name: 'description', content: 'Rosetta Stone' },
],
link: [
{ rel: 'prefetch', href: `/data-models/us/masthead/desktop/${this.id}.jpg?t=${Date.now()}`},
{ rel: 'prefetch', href: `/data-models/us/masthead/mobile/${this.id}.jpg?t=${Date.now()}`},
{
rel: 'prefetch', href:'//assets.adobedtm.com/289d54d757557d351111069aa8acc743e67b15e6/satelliteLib-7aef1085f7c59294e8610e2f691a22c48a200841.js',
as: 'script'
}
],
script: [
{ src: `/data-models/${process.env.locale == 'en' ? 'us' : process.env.locale}/js/products/${this.id}.js?t=${Date.now()}`, body: false },
{ src: '/nuxt-assets/js/vendor/jquery/jquery.min.js', body: true},
{
src: '//assets.adobedtm.com/289d54d757557d351111069aa8acc743e67b15e6/satelliteLib-7aef1085f7c59294e8610e2f691a22c48a200841.js',
body: true
},
//{ src: '/nuxt-assets/js/clock.js', body: true },
//{ src: '/nuxt-assets/js/crescendo.js', body: true }
]
}
},
/**
* We run this function on the server so we can pre-populate
* some data. We pre-populate the products view, so when page load the
* user sees products right away.
* < script src='rs.com/product-data/us/models/js/sitewide.json'>< /script>
*/
async asyncData ({ params, error, payload, $axios }) {
let model
// If we're hard reloading in dev mode, or,
// We're in dev mode and lost the model (became null on file save)
if (!process.client || (process.client && !model)) {
if (!payload) {
// This means user is in dev mode, and/or page is being hard reloaded.
//let modelURI = `https://rosettastone.com/lp/data-models/us/json/products/catalog.json`
let modelURI = `/data-models/us/json/products/catalog.json`
let json = await $axios.get(modelURI)
model = json.data
} else {
// Page is being generated. Use model passed as payload.model
model = payload.model
}
}
// Set page's state data here. Set on server, carries to client.
// route is passed by the generate function in nuxt.config.js
return {
id: model.id,
productList: model.products,
// The following ones can be overridden by url params
langcode: 'esp',
ribbonText: model.ribbontext,
offerText: model.offertext,
bonusMonth: model.bonusMonth,
fullPrice: model.fullprice,
productName: model.productnames,
xpay: model.xpay,
productFilter: model.productfilter || [ '3', '6', '12', '24']
}
},
/**
* Page state data.
* Some of these come prefilled from the server while pages are being
* generated, in the `asyncData()` method. And some of these can be
* overwritten via url parameters as well.
* The default values here is what you'll see while HMR'ing in dev mode.
*/
data() {
return {
locale: { ...locale },
// Holds name Page.id value (sitewide, sale, etc). Set on the server with $route
id: null,
// Our current 3-letter selected lang code. Initially set in mounted().
// Set via click, dropdown change, url param, etc
langcode: 'esp',
// Language object ( {name, region, code} )
language: null,
// List of products to show. Each time a language is changed, this
// gets populated and the product view gets updated (new cart urls, etc).
// It is set in the asyncData() method.
productList: [],
// Products we don't want to show. This is the default. Can be overwritten in asyncData()
productFilter: [ '3', '6', '12', '24'],
// If there's a special offer message, this will hold it.
mastheadOfferMessage: null,
// Holds RSI.expirationdate. Default to end of year.
expirationdate: 'December 31, 2020 23:59',
// Needed for the sole purpose of hiding/showing masthead components.
isMobile: null,
// Passed to <ProductBox /> - Bonus Months
//bonusMonths: {},
// Passed to <ProductBox />. Divide prices by x (only changes the html price)
// always used with showfull=1
//xPay: null,
// Passed to <ProductBox /> - Little banner on top of product box
//ribbonText: {},
// Passed to <ProductBox /> - Little banner in middle of product box
//offerText: {} //this.$route.query.offertext,
}
},
created () {
},
/**
* Called on client side.
* We use this to fill State data above
* before mount, since the /lp/globals/models/<model>.js
* has been loaded into the browser's window object.
* This essentially is our state data.
*/
async mounted() {
/*
this.$store.watch((state, getters) => state.langcode,
(newValue, oldValue) => {
this.products = this.productList.slice(0)
this.$forceUpdate()
})
*/
this.$store.subscribe((mutation, state) => {
if (mutation.type == 'UPDATE_LANGCODE') {
console.log('detected mutation')
this.productList = this.productList.slice(0)
this.$forceUpdate()
}
})
let defaultLangCode = this.$store.state.locale == 'en' ? 'esp' : 'eng'
this.$store.commit('UPDATE_LANGCODE', (this.$route.query.lang || defaultLangCode))
// Products gets rendered on server, but we update this so they can be rerendered in the client.
// We render them in the server so user doesn't see a FOUC, and if something goes wrong in client,
// page still shows some products user can add to cart.
//this.productList = window.pagedata.products
//this.expirationdate = window.pagedata.expirationDate || ''
this.isMobile = window.innerWidth <= 540
//this.bonusMonths = window.pagedata.bonusmonths
// This exposes state data to window. Maybe needed when using TnT, etc
window.vuedata = this
// Lazy load images
let imgs = document.querySelectorAll('[data-src], [data-bg]')
let observer = new IntersectionObserver(loadImg, {
// Start downloading when img is within 50px of Y axis
rootMargin: '50px 0px',
threshold: 0.01
});
// Start observing the images
imgs.forEach(img => observer.observe(img))
function loadImg (entries, observer)
{
entries.forEach(e => {
// If in viewport
let el = e.target
if (e.intersectionRatio > 0)
{
let bg = el.dataset.bg || null
let src = el.dataset.src || null
if (bg)
{
el.style.backgroundImage = `url('${bg}')`
}
if (src)
{
el.src = src
}
// Stop observing
observer.unobserve(e.target);
}
})
}
// If RSI has custom js or css, insert them
/*
if (window.pagedata.js) {
let script = document.createElement('script')
script.src = `/nuxt-assets/models/us/data/us/custom-js/${this.id}.js`
document.body.appendChild(script)
}
if (window.pagedata.css) {
let css = document.createElement('link')
css.rel = 'stylesheet'
css.type = 'text/css'
css.href = `/nuxt-assets/product-data/data/us/custom-css/${this.id}.css`
document.body.appendChild(css)
}
*/
},
watch: {
products: (n, o) => {
console.log('changed')
try {
console.log(`new: ${n[0].lang}`)
console.log(`old: ${o[0].lang}`)
} catch (e) {
}
}
},
computed: {
/**
* Everytime state's productList gets updated with a new value,
* this computed property will automatically be updated too.
* We have this computed property so we don't have to
* filter/map/sort the productsList each time we update it.
* Use this in the template view instead of productList
*/
products () {
console.log('-- productListe computed reran')
let p = this.productList
.filter(p => p.lang == this.$store.state.langcode)
.filter(p => this.productFilter.indexOf(p.lvl) != -1)
//.sort( (a, b) => parseInt(a.lvl) < parseInt(b.lvl) ? -1 : 1)
// TODO sort
if (this.isMobile) {
p = p.reverse()
}
return p
}
}
}
</script>
<style lang='stylus'>
body
margin 0
padding 0
font-family effra
//font-family gothamlight
.content
max-width 1150px
margin 0 auto
padding 0 1em
.masthead-offer
background $yellow
text-align center
padding 1.3em 0
font-size 1.3em
.masthead
background-color #333
background-size 100% 100%
padding 27% 0 0 0
line-height 1
box-shadow 0 4px 10px -6px inset
box-sizing border-box
&.desktop
@media $phone
display none
&.mobile
display none
@media $phone
display block
padding 36% 0% 1% 0%
.languages-wrap
//background #f1f1f1
padding-top 3em
.content
display flex
flex-direction row
justify-content center
& > div:first-of-type
align-self center
font-size 1.8em
padding-right 1em
@media (max-width:580px)
padding-top 2em
.content
flex-direction column
max-width 450px
& > div:first-of-type
font-size 1.8em
padding 0 0 20px 0
.products-container
position relative
margin-top 3em
//background #f1f1f1
p.disclaimer
margin 50px auto 20px
max-width 1000px
font-size 16px
font-family effra
&:after
content ''
position absolute
height 142%
width 40%
left 0
top -66px
background-image url(https://rosettastone.com/lp/common-modules/assets/modules_2018/gray-stone.svg)
background-repeat no-repeat
background-size contain
background-position center left
z-index -1
@media $phone
&:after
height 90%
width 100%
top -65px
left -100px
background-size cover
.products
display flex
flex-wrap wrap
justify-content space-between
@media $tablet
justify-content space-around
@media $phone
// Reverse product order in mobile
list = 1..15
for n in list
.product:nth-child({n})
order list[-(n)]
.highlights
padding 3em
position relative
h3
margin 0 0 2em 0
font-size 180%
text-align center
font-family helvetica
font-size 40px
.content > div
display flex
& > div
display flex
.bullets
flex-basis 50%
p
margin-bottom 3em
line-height 26px
ul
padding 0
margin 0
list-style none
li
margin-bottom 1em
background url('') no-repeat 0 12px
padding 10px 0 0 30px
.product-img
order 1
flex-basis 50%
background-repeat no-repeat
background-size contain
margin-left 5em
&:after
content ''
position absolute
height 107%
width 100%
right 0
top 7%
background-image url(https://rosettastone.com/lp/standard-page-assets/freetrial_assets/blue-stone.svg)
background-repeat no-repeat
background-size contain
background-position center right
z-index: -1
@media $tablet
.content
padding 0
& > div
flex-direction column
.bullets
order 1
font-size 18px
.product-img
order 0
min-height 310px
margin 0
background-position center
.company-quotes
padding 0 0 6em 0
h3
text-align center
@media $phone
.content
padding 0
.trusted
padding 5em 0
box-sizing border-box
position relative
& > div
text-align center
max-width 770px
h3
font-size 2.3em
margin-bottom 2em
font-family helvetica
.logos
display flex
align-items center
justify-content space-between
img:nth-of-type(1)
height 75px
img:nth-of-type(2)
height 50px
width 20%
img:nth-of-type(3)
height 65px
img:nth-of-type(4)
width 22%
&:after
content ''
position absolute
height 184%
width 100%
right 0
top 204px
background-image url(https://rosettastone.com/lp/common-modules/assets/modules_2018/gray-stone.svg)
background-repeat no-repeat
background-size contain
z-index: -1
transform scaleX(-1)
&:before
content ''
position absolute
height 184%
width 100%
left 0
top 204px
background-image url(https://rosettastone.com/lp/common-modules/assets/modules_2018/gray-stone.svg)
background-repeat no-repeat;
background-size contain;
background-position center left;
z-index -1
@media $tablet
padding-top 0
</style>