CTO of Collaboration App FigureOne

Consultant to Marketo agencies and clients

Sanford Whiteman

SLIGHTLY LESS PALEO

2014-Present

2011-2013

2001-2005

"That Forms and Munchkin Guy"

"That Reverse Proxy Guy"

"That IIS SMTP and IMail Guy"

"That Guy"

THE DISTANT PAST

Sysadmin (Windows-centric) and firewall admin (Cisco, CheckPoint)

Mail/messaging admin (Exchange, Notes, GroupWise, Postfix, sendmail) 

Back end and front end developer since 2005 (JS, PHP,  Java, C#, MSSQL, MySQL, IIS + great niche software)

MERELY VINTAGE

TODAY'S MESSAGE: BE A

 TECHNICAL TENANT

  • Sometimes, the greasiest wheel gets things done
  • Take out your frustration as community evangelism :)
  • SaaS providers want your knowledge, if you've done the research
  • It (sometimes) works in the real world, too!

MarSec (Marketing Security): an approach to online security that incorporates and increases brand value

Can rely on obscurity

Can mean awareness, not patching

Has customer-facing benefits

Increases compatibility

→ think "unguessable" links

→ if you know a form isn't secure, omit password field

→ SSL improves PageRank

and referrer visibility

→ SSL helps operate with broken image proxies

 MARSEC  vs. TRADITIONAL INFOSEC

  • Again, PageRank!
  • Respects your company's "SSL Everywhere" policy
  • See the referrer when the source uses SSL (Bing secure search and any other SSL site)
  • Adds real, live, no-joke brand protection
  • Protects against humiliating and/or costly phishing/tampering attacks

SSL (HTTPS) IS YOUR FRIEND

SSL DEPLOYMENT OPTIONS

  1. Upload cert to Marketo
    • ​​Onboarding cost ($1000+)
  2. Upload cert to your own CDN
    • Onboarding typically free
    • Monthly can be nearly free unless dedicated IP
    • Huge additional performance benefits

Not convinced? C'mon, it's fun to be scared...

  • http://branding.example.com can't use SSL from Marketo
  • Response from the click tracker can thus be tampered with
  • Easy for even a junior hacker, if you happen to be targeted
  • Only secure end-to-end with Marketo if you don't use click tracking (and who wants to do that?)
  • P.S. Bitly links won't use SSL unless you make sure to use https://bit.ly/

SSL:  THE CATCH!

  • Will it happen to me?  
    • Security guy: "It already has."
    • Person who wants to have friends: "Nah."
  • Truth: anyone can be hacked this way, but degree of targetude differs by company
  • The most powerful solution is to use a CDN

SSL: IS THE CATCH JUST PARANOIA?

WHAT IS  MUNCHKIN?

  • Marketo's version of Google Analytics (sort of!)
  • Principally for eventually-named leads
  • Automatically records full page views
  • Manually records synthetic page views and link clicks*
  • Manually associates named leads with one or more browser cookies (_mkto_tok)

MUNCHKIN COOKIE STRUCTURE


 _mkto_trk=id:410-XOR-673&token:_mch-jshell.net-1440118430418-75783
              | Munch ID|       |   Domain    | | Timestamp | |Rnd|
              +---------+       +-------------+ +-----------+ +---+                                       
  • Millisecond granularity + 5-digit random number = in practice, unique (in a web context)
  • A key-value sequence inside another key-value sequence (the document.cookie magic string)

MUNCHKIN COOKIE  RELATIONSHIPS

  • Can be reassociated to different leads over time, so lead activity log is mushy
  • One-to-many between a lead and its cookies
  • One-to-many between a cookie and its browsers
    • yes, browsers with an "s"!

  <a href="http://www.myotherdomain.com/?_mkto_trk={$_COOKIE['_mkto_trk']}">Let's sync up!</a>

WHAT MUNCHKIN LOGS

  • Web Page Visits (top-level documents) with Munchkin.init()(async)
  • Clicked Links that don't have the special class "mchNoDecorate" (sync w/timeout)
  • Synthetic Page Visits or Clicks via JS munchkinFunction() (async)
    • great for logging special events ("was on site for 15 pages," "watched video for 30 seconds")
    • you can't (easily) suppress the default VisitWebPage, though
    • also good for catching complex JS navigation

MUNCHKIN  RELIABILITY

  • Logs correctly the vast majority of the time
  • But lost metrics can be confounding, even if they don't change overall trends
  • When lost hits happen to appear in triggers, uh-oh!

ERROR DETECTION

  • Set-it-and-forget-it services are notoriously hard to troubleshoot
  • Startup options make Munchkin more reliable, but have you heard of them?
  • Advanced options can be more bane than boon

(Sorry, Elliott!)

Firefox's version

WHOA! WHAT JUST HAPPENED?

  • Technically, all we know is: "The browser stopped waiting for a response from the Marketo tracking server."
  • Don't know if the tracking call actually finished
  • Only the request part of the call is important, anyway (it's a GET)
  • Magic number 349/350/351ms tells us it was a timeout rather than a reset (reset can happen from the Mkto side, and looks messy but isn't a fail)

DIGGING  DEEPER

  • Put a proxy (Fiddler, Burp, James, etc.) in-between you and Marketo to see whether the call actually completed from Marketo's side
  • Usually it has completed
  • But watching the browser — deepest most users go! — you'd think you had rampant (as opposed to occasional) drops
  • Note proxy will still interpret server reset (RST) as failure, even if Marketo digests the data

A CONFLICT OF INTEREST

  • Ageless conflict between background importance in the browser and mission-critical importance to users
  • Logging unnecessary events (every link considered an exit link) complicates "importance"
  • Even the best browser feedback (Chrome Developer Tools) gives ambiguous results -- and few marketers are equipped to go deeper

BACKGROUND vs. FOREGROUND

  • Background tasks (also known as asychronous) are aborted when somebody goes to a new page
  • Foreground tasks (also known as synchronous) prevent the browser from doing anything else until they're done (or they time out)
  • So analytics has a bit of both: 
    • don't want to stop anyone from using our site... 
    • ... but don't want them to stop it from working!
  • Hmm.
  • Thus beacons.  But Munchkin doesn't use them yet.*

THE PROBLEM WITH  ClickLink

  • Absent other info, Munchkin assumes all links are exit links (incapable of logging at destination)
  • "Other info" (class mchNoDecorate) is scarcely documented
  • Default opt-in for click logging might be understandable
  • But Munchkin's click logging is synchronous and therefore holds up page unload by up to 350ms: very bad!

<>

<!-- 
this link goes to a page in our site that itself loads Munchkin, 
so there's no need to log the click 
-->
<a href="/some/other/page.html" class="mchNoDecorate">In-site link</a>

<!-- 
this link goes to a binary asset in our site, 
so we DO need to log the click
-->
<a href="/some/asset/like/a/whitepaper.pdf">In-site direct asset link</a>

<!-- 
this link goes to another site that doesn't load Munchkin 
(or doesn't load OUR Munchkin!) so log this one, too 
-->
<a href="http://www.offsite.com/someone/elses/page.html">Offsite (true exit) link</a>

THE SOLUTIONS

  • Intelligent use of mchNoDecorate
  • Deployment of Munchkin.beaconize() helper

WHAT'S THIS  "BEACONIZE"  YOU SPEAK OF?

  • A Munchkin helper script that adds support for the latest guaranteed background call technology, sendBeacon()
  • Supported only Firefox and Chrome
  • Pseudocode below... ask me for the code!

<>


# Pseudocode for this one!

HOOK ( ALL AJAX REQUESTS )
    IF ( AJAX URL IS aaa-bbb-ccc.mktoresp.com )
        IF ( BROWSER SUPPORTS SENDBEACON )
            COPY AJAX URL
            ABORT AJAX REQUEST
            SEND BEACON TO URL
        ELSE
            IGNORE
    ELSE
        IGNORE

        

BEACONS vs. ASYNC/SYNC AJAX

  • Sync call blocks navigation and other in-page behavior
  • Async call won't block, but will also not be guaranteed to finish before the user goes to a new page
  • Beacon won't block and will finish even after the user goes to a new page (and/or new website) or closes the current tab
  • Beacon usually sent immediately but may schedule for <1s later

MUNCHKIN AND  FORMS

  • Forms do not require Munchkin to work
    • embed a form anywhere without Munch
    • turn off Munch on a Marketo LP if you want
  • But Munchkin cookie won't be associated if not posted along with the form (doesn't mean the cookie was created on the page hosting the form — confusion reigns!)
  • cookieAnon option makes Munchkin start tracking only upon form submissions (not recommended)
  • Technically, _mkto_trk cookie becomes _mkt_trk field upon form post

THE  REFERRAL FORM  PROBLEM

  • You want a form that referrers can use to sign people up, without being reassociated to the latest new lead they sign up
  • It's relatively simple, yet takes more effort than people think
  • "Just remove Munchkin.js from the page" — nope!
  • You can't just remove Munchkin from the form, you have to archive (but not delete) the existing cookie, otherwise it'll be sent with the form post
  • Restore the cookie after the post
  • Thanks to the Forms 2.0 API, this is very easy

{}

MktoForms2.whenReady(function(form) {


		var munchkinCookieName = '_mkto_trk',
				munchkinCookieDomain = '.example.com',
				munchkinCookieValue;

		// prepare necessary dates for cookie management
		var BEGINNING_OF_EPOCH = 0,
				MUNCHKIN_COOKIE_LIFE_DAYS = 730, // 2 years
				tsToday = new Date(),
				tsDayZero = new Date(BEGINNING_OF_EPOCH).toGMTString(),
				tsMunchkinExpiry = new Date(tsToday.getTime() + 864E5 * MUNCHKIN_COOKIE_LIFE_DAYS).toGMTString();

		form.onSubmit(function(form) {

				// save the Munchin cookie's current value to be restored later
				munchkinCookieValue = (function getSingleCookie(name) {
						for (var cookies = document.cookie.split(/; ?/), i = 0, imax = cookies.length; i < imax; i++) {
								var cookie = cookies[i].split(/=/);
								if (cookie.shift() == name) return cookie.join();
						}
				})(munchkinCookieName);

				// delete the current cookie (set expiry in the past) so the on-the-wire form post will not include it
				document.cookie = munchkinCookieName + '=;path=/;domain=' + munchkinCookieDomain + ';expires=' + tsDayZero;
		});

		form.onSuccess(function(vals, thankYouURL) {

				// the form's been posted, now restore the cookie value we saved in the onsuccess handler
				document.cookie = munchkinCookieName + '=' + munchkinCookieValue + ';path=/;domain=' + munchkinCookieDomain + ';expires=' + tsMunchkinExpiry;

		})
})

THE  FORMS 2.0 API

  • Three simple event handlers: onValidate, onSubmit, onSuccess — but you can build almost anything
  • Getting to know the standard Forms 2.0 DOM is also very important
    • Wrappers around rows are unlabeled by default
    • Some CSS classes need to be removed when cloning elements
  • Don't use jQuery!

FORMS: COMPANY  TYPEAHEAD  W/GOOGLE

{}

MktoForms2.loadForm("//app-sj01.marketo.com", "410-XOR-673", 169,
function(form) 
{
    var formEl = form.getFormElem()[0];
    var companyField = formEl.querySelector('INPUT[name="Company"]');
    var websiteField = formEl.querySelector('INPUT[name="Website"]');    
    var firstNameField = formEl.querySelector('INPUT[name="FirstName"]');
    var emailField = formEl.querySelector('INPUT[name="Email"]');    

    var defaultBounds = new google.maps.LatLngBounds(
      new google.maps.LatLng(-90,-180),
      new google.maps.LatLng(90,180)
    );
    
    var options = {
        bounds: defaultBounds,
        types: ['establishment']
    };

    autocomplete = new google.maps.places.Autocomplete(companyField, options);
    
    google.maps.event.addListener(autocomplete, 'place_changed', function() {
    
        // shorten object to just name
        var acCompanyName = autocomplete.getPlace().name;
        companyField.value = acCompanyName;    
        
        // just for fun, fill website too
        var acWebsite = autocomplete.getPlace().website;
        if (acWebsite) websiteField.value = acWebsite;   
        
        // additional interesting types    
        var address_components = autocomplete.getPlace().address_components,
            acAddress, 
            acCity, 
            acState;
        
        for ( var i = 0, imax = address_components.length; i < imax; i++ ) { 
            var component = address_components[i], 
                types = component.types, 
                short_name = component.short_name,
                long_name = component.long_name;
        
            for ( var ii = 0, iimax = types.length; ii < iimax; ii++ ) {
                var type = types[ii];
                switch (type) {
                case 'administrative_area_level_1':
                    acState = short_name;
                    break;
                case 'locality':
                case 'sublocality':
                    acCity = long_name;
                    break;
                case 'route':
                    acAddress  = !acAddress ? short_name : acAddress + ' ' + short_name;
                    break;
                case 'street_number':
                    acAddress = !acAddress ? short_name : short_name + ' ' + acAddress;
                    break;
                default:
                }
    
            }
        }
        
        console.log( acAddress, acCity, acState );

    });    
});  

FORMS:  VALIDATE AN EMAIL  IN REAL-TIME

{}

MktoForms2.loadForm("//app-sj01.marketo.com", "410-XOR-673", 341)
MktoForms2.whenReady(function(form) {

		var emailExtendedValidationError = 'Doesn\'t sound like a real TLD!',
				emailField = 'Email',
				emailFieldSelector = '#' + emailField,
				extendedValidationComplete = false;

		form.onValidate(function(builtInValidationComplete) {

				if (builtInValidationComplete) {
						form.submittable(extendedValidationComplete); // we'll need to reenable on cb either way

						if (!form.submittable()) {
								var currentTLD = form.vals()[emailField].split('.').pop();

								dnsDomainExists(currentTLD, function(exists) {
										if (!exists) {
												form.showErrorMessage(emailExtendedValidationError, form.getFormElem().find(emailFieldSelector));
										} else {
												extendedValidationComplete = true;
												form.submit();
										}
								});
						}
				}
		});

		function dnsDomainExists(dnsDomain, callback) {

				var dnsAPIURL = 'http://api.statdns.com/' + dnsDomain + '/ns ',
						xdr;

				if (window.XDomainRequest && !('withCredentials' in window.XMLHttpRequest.prototype)) {
						xdr = new XDomainRequest;
						xdr.open('GET', dnsAPIURL); // IE8: HEAD + other peccadilloes
				} else {
						xdr = new XMLHttpRequest;
						xdr.open('HEAD', dnsAPIURL);
				}

				xdr.onerror = function() {
						if (xdr instanceof XMLHttpRequest) {
								alert('Unhandled error');
						} else {
								console.log('XDR handled error: API returned failure');
								callback(false);
						}

				};

				xdr.onload = function() {
						if (xdr.status == 200) {
								console.log('XHR handled success: API returned success');
								callback(true);
						} else if (xdr instanceof XMLHttpRequest) {
								console.log('XHR handled success: API returned failure');
								callback(false);
						} else {
								console.log('XDR handled success: API returned success');
								callback(true);
						}
				};

				xdr.send();
		}

});

FORMS: UPLOAD AN  IMAGE!

Heck yes, it's cool.  But make a fallback for IE 8 & 9.

{}

MktoForms2.loadForm("//app-sj01.marketo.com", "410-XOR-673", 341,
    function(form) 
    {    
        var fieldToPopulate = 'MarketoSocialLinkedInProfileURL'; // the Forms 2.0 field name that will hold the image data

        var MAX_WIDTH = 200, MAX_HEIGHT = 200, formValues = {};    
    
        // dynamically add a file input; Forms 2.0 won't know about it directly but we'll use it to push image data into the form
        var profileImageFile = document.createElement('INPUT');
        profileImageFile.id = 'profileImageFile';
        profileImageFile.type = 'file';        
        var existingProfileField = document.getElementById(fieldToPopulate);
        existingProfileField.parentNode.insertBefore( profileImageFile, existingProfileField);
        
        // chain events: user selects image, draw image data on a dynamic CANVAS el to resize, then extract as Base64 and add to Marketo form
        profileImageFile.addEventListener('change', function(changeEvent)
        {    
            var reader = new FileReader();
            reader.addEventListener('loadend',function(progressEvent)
            { 
                var originalImg = new Image();
                originalImg.addEventListener('load', function() 
                {
                    var canvas = document.createElement('canvas');
    
                    // resize to fit max dimensions
                    if ( originalImg.width > MAX_WIDTH && originalImg.width >= originalImg.height ) {
                        canvas.width = MAX_WIDTH, canvas.height = MAX_WIDTH * originalImg.height / originalImg.width;
                    } else if ( originalImg.height > MAX_HEIGHT ) {
                        canvas.height = MAX_HEIGHT, canvas.width = MAX_HEIGHT * originalImg.width / originalImg.height;
                    } else {
                        canvas.width = originalImg.width, canvas.height = originalImg.height;
                    }
                    
                    // draw onto a purposely-sized canvas to reduce size
                    canvas.getContext('2d').drawImage(originalImg, 0,0, canvas.width, canvas.height);
                    
                    // extract contents of canvas
                    var resizedImgData = canvas.toDataURL('image/jpeg');
                    
                    // now write (string) data to Forms 2.0 field
                    formValues[fieldToPopulate] = resizedImgData;
                    form.setValues(formValues);
    
                    // display resized image for fun
                    var lastPreviewImg = existingProfileField.parentNode.querySelector('.imagePreview');
                    !lastPreviewImg || existingProfileField.parentNode.removeChild( lastPreviewImg );

                    var newPreviewImg = new Image();
                    newPreviewImg.src = resizedImgData;
                    newPreviewImg.className = 'imagePreview';    
                    existingProfileField.parentNode.insertBefore( newPreviewImg, existingProfileField );
                });
                originalImg.src = reader.result;        
            });
            reader.readAsDataURL(changeEvent.target.files[0]);
        });
    });        

FORMS:  TRANSLATE CONTENT  ON THE FLY

{}

MktoForms2.loadForm("//app-sj01.marketo.com", "410-XOR-673", 298)
MktoForms2.whenReady(function(form) {

		var formEl = form.getFormElem()[0],
            commentsField = 'Comments__c',
            googleTranslateAPIBase = 'https://www.googleapis.com/language/translate/v2',
            googleTranslateSource = 'en',
            googleTranslateTarget = 'de',
            googleTranslateAPIKey = 'AIzaSyC9vOmwqqRq-UfunFilWcYqIf1fTPg0rik',
            validationComplete = false;
    
        // Sandy's demo email
        		form.setValues({Email:'jtestoro@vaneckdemos.com'});                                 
        formEl.querySelector('#Email').readOnly = true;
    
        form.onValidate(function(valid) {

            form.submittable(validationComplete); // reenable on cb

            if (valid && !form.submittable()) {
                
                var originalValues = form.getValues(),
                    originalComments = originalValues[commentsField],
                    translatedValues = {};
                
                simpleJSONP( googleTranslateAPIBase + '?'
                             + '&source=' + googleTranslateSource
                             + '&target=' + googleTranslateTarget
                             + '&key=' + googleTranslateAPIKey
                             + '&q=' + encodeURIComponent(originalComments),
                             window.translateComments = function translateComments(response) {
                                translatedValues[commentsField] = response.data.translations[0].translatedText;
                                form.setValues(translatedValues);
                                validationComplete = true;
                                form.submit();
                             });
            } else if (valid) {            
                form.submittable(false); // remove this line on your production site, but leave it in for the demo!
            }
		});
    
		function simpleJSONP(url,callback) 
        {
          var scriptEl = document.createElement('script');
          scriptEl.type = 'text/javascript';
          scriptEl.src = url + '&callback=' + callback.name;                      
          document.querySelector('HEAD').appendChild(scriptEl);                        
		}
        

           

});

FORMS:  LOCALIZE  A SINGLE FORM

?lang=en-US

?lang=fr-FR

<>

The DOM is strong with this one!

... and introducing the Data Transfer Object

{}

Munchkin.init('410-XOR-673');
MktoForms2.loadForm("//app-sj01.marketo.com", "410-XOR-673", 268,
		function(form) {

				// utility  fn: tag outer wrapper DIVs with their inner inputs' names to make them more findable
				for (var formEl = form.getFormElem()[0], formRows = formEl.querySelectorAll('.mktoFormRow'), i = 0, imax = formRows.length, wrappedField; i < imax; i++) {
						if ((wrappedField = formRows[i].querySelector('INPUT,SELECT,TEXTAREA')) && wrappedField.name) {
								formRows[i].setAttribute('data-wrapper-for', wrappedField.name);
						}
				}

				// get current language setting (passed in URL or cookie or ... )
				var currentLang = form.getValues().Media_Language__c;

				// this is just for the cosmetic flag
				formEl.setAttribute('lang', currentLang);

				// get translations embedded in form rich text area
				var translateMapEl = formEl.querySelector('#translateMap'),
						translateMap = JSON.parse(translateMapEl.textContent || translateMapEl.innerText),
						currentTranslations = translateMap[currentLang];

				// loop over translatables
				var fieldName, fieldTranslations, propertyName, propertyValue, translateTargetEl;
				for (fieldName in currentTranslations) {
						fieldTranslations = currentTranslations[fieldName];
						for (propertyName in fieldTranslations) {
								propertyValue = fieldTranslations[propertyName];
								switch (propertyName) {
										case "label":
												translateTargetEl = formEl.querySelector('[data-wrapper-for="' + fieldName + '"] LABEL');
												document.documentElement.innerText ? translateTargetEl.lastChild.nodeValue = propertyValue : translateTargetEl.lastChild.textContent = propertyValue;
												break;
										case "validation":
												translateTargetEl = MktoForms2.$(formEl.querySelector('[data-wrapper-for="' + fieldName + '"] .mktoFieldDescriptor'));
												translateTargetEl.data('mktoFieldDescriptor').validationMessage = propertyValue;
												break;
										case "placeholder":
												translateTargetEl = formEl.querySelector('[data-wrapper-for="' + fieldName + '"] INPUT');
												translateTargetEl.setAttribute('placeholder', propertyValue);
												break;
										case "submit":
												translateTargetEl = formEl.querySelector('BUTTON[type="submit"]');
												translateTargetEl.innerHTML = propertyValue;
												break;
								}
						}
				}

		});

<>

/* hide the last rich text div used to pass JSON translate map */

.mktoFormRow:not([data-wrapper-for]) {
		display: none;
}

#translateMap
/* IE8 */

{
		display: none;
}

.mktoForm {
		margin: 20px;
		padding: 20px;
		border: 1px solid gray;
		border-radius: 6px;
	  background-color: beige;
}

.mktoForm.flag:after {
		position: relative;
		left: 8px;
		content: url('');
		background-image: url('');
}

.mktoForm.flag:lang(ad):after {
		background-position: -16px 0
}

.mktoForm.flag:lang(ae):after {
		background-position: -32px 0
}

.mktoForm.flag:lang(af):after {
		background-position: -48px 0
}

.mktoForm.flag:lang(ag):after {
		background-position: -64px 0
}

.mktoForm.flag:lang(ai):after {
		background-position: -80px 0
}

.mktoForm.flag:lang(al):after {
		background-position: -96px 0
}

.mktoForm.flag:lang(am):after {
		background-position: -112px 0
}

.mktoForm.flag:lang(an):after {
		background-position: -128px 0
}

.mktoForm.flag:lang(ao):after {
		background-position: -144px 0
}

.mktoForm.flag:lang(ar):after {
		background-position: -160px 0
}

.mktoForm.flag:lang(as):after {
		background-position: -176px 0
}

.mktoForm.flag:lang(at):after {
		background-position: -192px 0
}

.mktoForm.flag:lang(au):after {
		background-position: -208px 0
}

.mktoForm.flag:lang(aw):after {
		background-position: -224px 0
}

.mktoForm.flag:lang(az):after {
		background-position: 0 -16px
}

.mktoForm.flag:lang(ba):after {
		background-position: -16px -16px
}

.mktoForm.flag:lang(bb):after {
		background-position: -32px -16px
}

.mktoForm.flag:lang(bd):after {
		background-position: -48px -16px
}

.mktoForm.flag:lang(be):after {
		background-position: -64px -16px
}

.mktoForm.flag:lang(bf):after {
		background-position: -80px -16px
}

.mktoForm.flag:lang(bg):after {
		background-position: -96px -16px
}

.mktoForm.flag:lang(bh):after {
		background-position: -112px -16px
}

.mktoForm.flag:lang(bi):after {
		background-position: -128px -16px
}

.mktoForm.flag:lang(bj):after {
		background-position: -144px -16px
}

.mktoForm.flag:lang(bm):after {
		background-position: -160px -16px
}

.mktoForm.flag:lang(bn):after {
		background-position: -176px -16px
}

.mktoForm.flag:lang(bo):after {
		background-position: -192px -16px
}

.mktoForm.flag:lang(br):after {
		background-position: -208px -16px
}

.mktoForm.flag:lang(bs):after {
		background-position: -224px -16px
}

.mktoForm.flag:lang(bt):after {
		background-position: 0 -32px
}

.mktoForm.flag:lang(bw):after {
		background-position: -16px -32px
}

.mktoForm.flag:lang(by):after {
		background-position: -32px -32px
}

.mktoForm.flag:lang(bz):after {
		background-position: -48px -32px
}

.mktoForm.flag:lang(ca):after {
		background-position: -64px -32px
}

.mktoForm.flag:lang(cd):after {
		background-position: -80px -32px
}

.mktoForm.flag:lang(cf):after {
		background-position: -96px -32px
}

.mktoForm.flag:lang(cg):after {
		background-position: -112px -32px
}

.mktoForm.flag:lang(ch):after {
		background-position: -128px -32px
}

.mktoForm.flag:lang(ci):after {
		background-position: -144px -32px
}

.mktoForm.flag:lang(ck):after {
		background-position: -160px -32px
}

.mktoForm.flag:lang(cl):after {
		background-position: -176px -32px
}

.mktoForm.flag:lang(cm):after {
		background-position: -192px -32px
}

.mktoForm.flag:lang(cn):after {
		background-position: -208px -32px
}

.mktoForm.flag:lang(co):after {
		background-position: -224px -32px
}

.mktoForm.flag:lang(cr):after {
		background-position: 0 -48px
}

.mktoForm.flag:lang(cu):after {
		background-position: -16px -48px
}

.mktoForm.flag:lang(cv):after {
		background-position: -32px -48px
}

.mktoForm.flag:lang(cy):after {
		background-position: -48px -48px
}

.mktoForm.flag:lang(cz):after {
		background-position: -64px -48px
}

.mktoForm.flag:lang(de):after {
		background-position: -80px -48px
}

.mktoForm.flag:lang(dj):after {
		background-position: -96px -48px
}

.mktoForm.flag:lang(dk):after {
		background-position: -112px -48px
}

.mktoForm.flag:lang(dm):after {
		background-position: -128px -48px
}

.mktoForm.flag:lang(do):after {
		background-position: -144px -48px
}

.mktoForm.flag:lang(dz):after {
		background-position: -160px -48px
}

.mktoForm.flag:lang(ec):after {
		background-position: -176px -48px
}

.mktoForm.flag:lang(ee):after {
		background-position: -192px -48px
}

.mktoForm.flag:lang(eg):after {
		background-position: -208px -48px
}

.mktoForm.flag:lang(eh):after {
		background-position: -224px -48px
}

.mktoForm.flag:lang(er):after {
		background-position: 0 -64px
}

.mktoForm.flag:lang(es):after {
		background-position: -16px -64px
}

.mktoForm.flag:lang(et):after {
		background-position: -32px -64px
}

.mktoForm.flag:lang(fi):after {
		background-position: -48px -64px
}

.mktoForm.flag:lang(fj):after {
		background-position: -64px -64px
}

.mktoForm.flag:lang(fm):after {
		background-position: -80px -64px
}

.mktoForm.flag:lang(fo):after {
		background-position: -96px -64px
}

.mktoForm.flag:lang(fr):after,
.mktoForm.flag:lang(fr-FR):after {
		background-position: -112px -64px
}

.mktoForm.flag:lang(ga):after {
		background-position: -128px -64px
}

.mktoForm.flag:lang(gb):after,
.mktoForm.flag:lang(en-GB):after {
		background-position: -144px -64px
}

.mktoForm.flag:lang(gd):after {
		background-position: -160px -64px
}

.mktoForm.flag:lang(ge):after {
		background-position: -176px -64px
}

.mktoForm.flag:lang(gg):after {
		background-position: -192px -64px
}

.mktoForm.flag:lang(gh):after {
		background-position: -208px -64px
}

.mktoForm.flag:lang(gi):after {
		background-position: -224px -64px
}

.mktoForm.flag:lang(gl):after {
		background-position: 0 -80px
}

.mktoForm.flag:lang(gm):after {
		background-position: -16px -80px
}

.mktoForm.flag:lang(gn):after {
		background-position: -32px -80px
}

.mktoForm.flag:lang(gp):after {
		background-position: -48px -80px
}

.mktoForm.flag:lang(gq):after {
		background-position: -64px -80px
}

.mktoForm.flag:lang(gr):after {
		background-position: -80px -80px
}

.mktoForm.flag:lang(gt):after {
		background-position: -96px -80px
}

.mktoForm.flag:lang(gu):after {
		background-position: -112px -80px
}

.mktoForm.flag:lang(gw):after {
		background-position: -128px -80px
}

.mktoForm.flag:lang(gy):after {
		background-position: -144px -80px
}

.mktoForm.flag:lang(hk):after {
		background-position: -160px -80px
}

.mktoForm.flag:lang(hn):after {
		background-position: -176px -80px
}

.mktoForm.flag:lang(hr):after {
		background-position: -192px -80px
}

.mktoForm.flag:lang(ht):after {
		background-position: -208px -80px
}

.mktoForm.flag:lang(hu):after {
		background-position: -224px -80px
}

.mktoForm.flag:lang(id):after {
		background-position: 0 -96px
}

.mktoForm.flag:lang(ie):after {
		background-position: -16px -96px
}

.mktoForm.flag:lang(il):after {
		background-position: -32px -96px
}

.mktoForm.flag:lang(im):after {
		background-position: -48px -96px
}

.mktoForm.flag:lang(in):after {
		background-position: -64px -96px
}

.mktoForm.flag:lang(iq):after {
		background-position: -80px -96px
}

.mktoForm.flag:lang(ir):after {
		background-position: -96px -96px
}

.mktoForm.flag:lang(is):after {
		background-position: -112px -96px
}

.mktoForm.flag:lang(it):after {
		background-position: -128px -96px
}

.mktoForm.flag:lang(je):after {
		background-position: -144px -96px
}

.mktoForm.flag:lang(jm):after {
		background-position: -160px -96px
}

.mktoForm.flag:lang(jo):after {
		background-position: -176px -96px
}

.mktoForm.flag:lang(jp):after {
		background-position: -192px -96px
}

.mktoForm.flag:lang(ke):after {
		background-position: -208px -96px
}

.mktoForm.flag:lang(kg):after {
		background-position: -224px -96px
}

.mktoForm.flag:lang(kh):after {
		background-position: 0 -112px
}

.mktoForm.flag:lang(ki):after {
		background-position: -16px -112px
}

.mktoForm.flag:lang(km):after {
		background-position: -32px -112px
}

.mktoForm.flag:lang(kn):after {
		background-position: -48px -112px
}

.mktoForm.flag:lang(kp):after {
		background-position: -64px -112px
}

.mktoForm.flag:lang(kr):after {
		background-position: -80px -112px
}

.mktoForm.flag:lang(kw):after {
		background-position: -96px -112px
}

.mktoForm.flag:lang(ky):after {
		background-position: -112px -112px
}

.mktoForm.flag:lang(kz):after {
		background-position: -128px -112px
}

.mktoForm.flag:lang(la):after {
		background-position: -144px -112px
}

.mktoForm.flag:lang(lb):after {
		background-position: -160px -112px
}

.mktoForm.flag:lang(lc):after {
		background-position: -176px -112px
}

.mktoForm.flag:lang(li):after {
		background-position: -192px -112px
}

.mktoForm.flag:lang(lk):after {
		background-position: -208px -112px
}

.mktoForm.flag:lang(lr):after {
		background-position: -224px -112px
}

.mktoForm.flag:lang(ls):after {
		background-position: 0 -128px
}

.mktoForm.flag:lang(lt):after {
		background-position: -16px -128px
}

.mktoForm.flag:lang(lu):after {
		background-position: -32px -128px
}

.mktoForm.flag:lang(lv):after {
		background-position: -48px -128px
}

.mktoForm.flag:lang(ly):after {
		background-position: -64px -128px
}

.mktoForm.flag:lang(ma):after {
		background-position: -80px -128px
}

.mktoForm.flag:lang(mc):after {
		background-position: -96px -128px
}

.mktoForm.flag:lang(md):after {
		background-position: -112px -128px
}

.mktoForm.flag:lang(me):after {
		background-position: -128px -128px
}

.mktoForm.flag:lang(mg):after {
		background-position: -144px -128px
}

.mktoForm.flag:lang(mh):after {
		background-position: -160px -128px
}

.mktoForm.flag:lang(mk):after {
		background-position: -176px -128px
}

.mktoForm.flag:lang(ml):after {
		background-position: -192px -128px
}

.mktoForm.flag:lang(mm):after {
		background-position: -208px -128px
}

.mktoForm.flag:lang(mn):after {
		background-position: -224px -128px
}

.mktoForm.flag:lang(mo):after {
		background-position: 0 -144px
}

.mktoForm.flag:lang(mq):after {
		background-position: -16px -144px
}

.mktoForm.flag:lang(mr):after {
		background-position: -32px -144px
}

.mktoForm.flag:lang(ms):after {
		background-position: -48px -144px
}

.mktoForm.flag:lang(mt):after {
		background-position: -64px -144px
}

.mktoForm.flag:lang(mu):after {
		background-position: -80px -144px
}

.mktoForm.flag:lang(mv):after {
		background-position: -96px -144px
}

.mktoForm.flag:lang(mw):after {
		background-position: -112px -144px
}

.mktoForm.flag:lang(mx):after {
		background-position: -128px -144px
}

.mktoForm.flag:lang(my):after {
		background-position: -144px -144px
}

.mktoForm.flag:lang(mz):after {
		background-position: -160px -144px
}

.mktoForm.flag:lang(na):after {
		background-position: -176px -144px
}

.mktoForm.flag:lang(nc):after {
		background-position: -192px -144px
}

.mktoForm.flag:lang(ne):after {
		background-position: -208px -144px
}

.mktoForm.flag:lang(ng):after {
		background-position: -224px -144px
}

.mktoForm.flag:lang(ni):after {
		background-position: 0 -160px
}

.mktoForm.flag:lang(nl):after {
		background-position: -16px -160px
}

.mktoForm.flag:lang(no):after {
		background-position: -32px -160px
}

.mktoForm.flag:lang(np):after {
		background-position: -48px -160px
}

.mktoForm.flag:lang(nr):after {
		background-position: -64px -160px
}

.mktoForm.flag:lang(nz):after {
		background-position: -80px -160px
}

.mktoForm.flag:lang(om):after {
		background-position: -96px -160px
}

.mktoForm.flag:lang(pa):after {
		background-position: -112px -160px
}

.mktoForm.flag:lang(pe):after {
		background-position: -128px -160px
}

.mktoForm.flag:lang(pf):after {
		background-position: -144px -160px
}

.mktoForm.flag:lang(pg):after {
		background-position: -160px -160px
}

.mktoForm.flag:lang(ph):after {
		background-position: -176px -160px
}

.mktoForm.flag:lang(pk):after {
		background-position: -192px -160px
}

.mktoForm.flag:lang(pl):after {
		background-position: -208px -160px
}

.mktoForm.flag:lang(pr):after {
		background-position: -224px -160px
}

.mktoForm.flag:lang(ps):after {
		background-position: 0 -176px
}

.mktoForm.flag:lang(pt):after {
		background-position: -16px -176px
}

.mktoForm.flag:lang(pw):after {
		background-position: -32px -176px
}

.mktoForm.flag:lang(py):after {
		background-position: -48px -176px
}

.mktoForm.flag:lang(qa):after {
		background-position: -64px -176px
}

.mktoForm.flag:lang(re):after {
		background-position: -80px -176px
}

.mktoForm.flag:lang(ro):after {
		background-position: -96px -176px
}

.mktoForm.flag:lang(rs):after {
		background-position: -112px -176px
}

.mktoForm.flag:lang(ru):after {
		background-position: -128px -176px
}

.mktoForm.flag:lang(rw):after {
		background-position: -144px -176px
}

.mktoForm.flag:lang(sa):after {
		background-position: -160px -176px
}

.mktoForm.flag:lang(sb):after {
		background-position: -176px -176px
}

.mktoForm.flag:lang(sc):after {
		background-position: -192px -176px
}

.mktoForm.flag:lang(sd):after {
		background-position: -208px -176px
}

.mktoForm.flag:lang(se):after {
		background-position: -224px -176px
}

.mktoForm.flag:lang(sg):after {
		background-position: 0 -192px
}

.mktoForm.flag:lang(si):after {
		background-position: -16px -192px
}

.mktoForm.flag:lang(sk):after {
		background-position: -32px -192px
}

.mktoForm.flag:lang(sl):after {
		background-position: -48px -192px
}

.mktoForm.flag:lang(sm):after {
		background-position: -64px -192px
}

.mktoForm.flag:lang(sn):after {
		background-position: -80px -192px
}

.mktoForm.flag:lang(so):after {
		background-position: -96px -192px
}

.mktoForm.flag:lang(sr):after {
		background-position: -112px -192px
}

.mktoForm.flag:lang(st):after {
		background-position: -128px -192px
}

.mktoForm.flag:lang(sv):after {
		background-position: -144px -192px
}

.mktoForm.flag:lang(sy):after {
		background-position: -160px -192px
}

.mktoForm.flag:lang(sz):after {
		background-position: -176px -192px
}

.mktoForm.flag:lang(tc):after {
		background-position: -192px -192px
}

.mktoForm.flag:lang(td):after {
		background-position: -208px -192px
}

.mktoForm.flag:lang(tg):after {
		background-position: -224px -192px
}

.mktoForm.flag:lang(th):after {
		background-position: 0 -208px
}

.mktoForm.flag:lang(tj):after {
		background-position: -16px -208px
}

.mktoForm.flag:lang(tl):after {
		background-position: -32px -208px
}

.mktoForm.flag:lang(tm):after {
		background-position: -48px -208px
}

.mktoForm.flag:lang(tn):after {
		background-position: -64px -208px
}

.mktoForm.flag:lang(to):after {
		background-position: -80px -208px
}

.mktoForm.flag:lang(tr):after {
		background-position: -96px -208px
}

.mktoForm.flag:lang(tt):after {
		background-position: -112px -208px
}

.mktoForm.flag:lang(tv):after {
		background-position: -128px -208px
}

.mktoForm.flag:lang(tw):after {
		background-position: -144px -208px
}

.mktoForm.flag:lang(tz):after {
		background-position: -160px -208px
}

.mktoForm.flag:lang(ua):after {
		background-position: -176px -208px
}

.mktoForm.flag:lang(ug):after {
		background-position: -192px -208px
}

.mktoForm.flag:lang(us):after,
.mktoForm.flag:lang(en-US):after,
.mktoForm.flag:lang(en-EN):after {
		background-position: -208px -208px
}

.mktoForm.flag:lang(uy):after {
		background-position: -224px -208px
}

.mktoForm.flag:lang(uz):after {
		background-position: 0 -224px
}

.mktoForm.flag:lang(va):after {
		background-position: -16px -224px
}

.mktoForm.flag:lang(vc):after {
		background-position: -32px -224px
}

.mktoForm.flag:lang(ve):after {
		background-position: -48px -224px
}

.mktoForm.flag:lang(vg):after {
		background-position: -64px -224px
}

.mktoForm.flag:lang(vi):after {
		background-position: -80px -224px
}

.mktoForm.flag:lang(vn):after {
		background-position: -96px -224px
}

.mktoForm.flag:lang(vu):after {
		background-position: -112px -224px
}

.mktoForm.flag:lang(ws):after {
		background-position: -128px -224px
}

.mktoForm.flag:lang(ye):after {
		background-position: -144px -224px
}

.mktoForm.flag:lang(za):after {
		background-position: -160px -224px
}

.mktoForm.flag:lang(zm):after {
		background-position: -176px -224px
}

.mktoForm.flag:lang(zw):after {
		background-position: -192px -224px
}

KEY TECH:  DATA TRANSFER OBJECT (DTO)


 <div id="translateMap">
 {
		"en-US": {
				"LastName": {
						"label": "Surname",
						"validation": "Make sure that's your real last name.",
						"submit": "Go"
				}
		},
		"fr-FR": {
				"FirstName": {
						"label": "Prénom",
						"placeholder": "Victor",
						"submit": "Envoi"
				}
		}
 }
 </div>

Nothing more than a hidden DIV inside a Rich Text field —

but the possibilities are endless!

THE  POLITICS OF PREFILL

  • Users "authenticate" using only email address
  • "Data at Rest" (DAR) concern when form fields are thought to be in motion only
  • Explicit rules and regulations (PII)
  • Creepiness factor
  • But still nice to have a choice!

PREFILL ON  EMBEDDED  FORMS

  • On Marketo LPs, field values are written into the HTML body as a JS object, then used to populate form inputs
  • What if we could get them into external pages using... the same method? (Think DTO. Think hard.)
  • Maybe I'll tell you how after the Q&A!

<>

    <div id="JSON_payload" data-type="application/json">
        { 
      		"FirstName" : "{{Lead.First Name}}" ,
      		"LastName" : "{{Lead.Last Name}}" ,
      		"Email" : "{{lead.Email Address}}" ,
      		"Website" : "{{company.Website}}" ,
      		"IP" : "{{lead.Anonymous IP}}"
      	}
    </div>

NYC MUG 2015-08 v2

By Sanford Whiteman, TEKNKL

Private

NYC MUG 2015-08 v2