Unified Shipping Module in an iframe

For security-critical environments, the widget can also be nested in an iframe

Callback Triggers

Entered PostCode

Selected Shipping option:

Type:

Data:

Usage

Adding the element

To integrate the consolidated checkout as an iframe in a page, define an iframe loading the widget iframe integration page. The iframe needs an id to be able to reference it from JS in order to send messages to it. As the height of the consolidated checkout can change dynamically during usage, an event is sent to the parent frame on height changes, so the iframe height can be adjusted accordingly.

<iframe
  title="checkoutIFrame"
  id="checkoutIFrame"
  style="width: 100%"
  src="https://widget.porterbuddy-test.com/porterbuddy-checkout-frame.html">
</iframe>

Communication with the iframe

All communication with the iframe is performed via window events. To send events to the widget iframe, use the the function "window.postMessage" on the contentWindow property of the iframe element. The unified shipping module iframe will respond with events that can be received by registering an event listener for the event type "message".

// send an event to the iframe
iframeWindow = document.getElementById("checkoutIFrame").contentWindow;

iframeWindow.postMessage({
  action: "pb-force-refresh",
  payload: options
}, "https://widget.porterbuddy-test.com");

// ... other events in similar fashion

// receive events from the iframe
window.addEventListener("message", function(event) {
// verify that the event came from the iframe page to secure against potential event injections
  if (event.origin = 'https://widget.porterbuddy-test.com' && event.data && event.data.action) {
    switch(event.data.action) {
      case 'pb-frame-ready': {
        // iframe content has loaded, initialize checkout.
        iframeWindow = document.getElementById("checkoutIFrame").contentWindow;
        iframeWindow.postMessage({
          action: "pb-checkout-init",
          payload: options
        }, "https://widget.porterbuddy-test.com");
        break;
      }
      case 'pb-on-selection-changed': {
        shippingSelection = event.data.payload;
        // ... e.g. store changed shipping selection
        break;
      }
      case 'pb-on-height-changed': {
        // change iframe height to render height from iframe content + margins
        document.getElementById('checkoutIFrame').style.height = (value + 30) + 'px';
        break;
      }
      // handle more events in similar fashion
      default:
      // nothing to do here
    }
  }
});

Events

All events to and from the consolidated checkout iframe are using this interface:

{
  action: string;
  payload: any;
}


Events to the consolidated checkout iframe

Event Payload Type Description
pb-checkout-init
checkoutOptions
Initialized the consolidated checkout in the iframe with the passed options. Available are all properties that are not functions or the "selectionPropertyChangeListener" property. To provide a similar functionality, an additional property "monitoredProperties" is supported.
pb-force-refresh
null | checkoutOptions
Sets the options to the new values and refreshes the whole widget context. If an option object is passed, it must be the whole object
pb-set-recipient-info
RecipientInfo
Sets the recipient info (postCode and optional email) for the checkout
pb-update-shipping-options
all shipping option arrays + postCodeEditable property
Updates the shipping options displayed and optionally enables / disables the postcode input


Events from the consolidated checkout iframe

Event Payload Type Description
pb-on-selection-changed
{
	selectedCategory: ShippingCategory,
	selectedShipping: SelectedShipping
}
Event that indicates that the shipping selection has changed
pb-on-unselect-shipping
null
Event to indicate that no shipping option is currently selected
pb-on-postcode-entered
string
Event to indicate that a postcode was entered. Contains the postcode as payload
pb-on-selected-property-changed
{
	optionId: string,
	propertyPath: string
	value: any (type of the changed prperty)
}
Event to indicate that a property from the monitoredProperties list has changed
pb-on-height-changed
number
Event that is thrown when the rendered height of the consolidated checkout widget changes. The payload contains the height in pixes without any outside margins

Integration hints

Initialization
As the page in the iframe loads resources, sending the initialization event immediately is in danger of the page being not loaded completely and the event getting lost. To make sure the widget iframe is ready to receive events, the widget page sends the event "pb-frame-ready" to it's paraent once the event listener is registered. Best practice is to send the initialization event from the main page once this event is received.

Security
As recommended by the documentation for "window.postMessage", the target origin for messages should be explicitly specified, and the origin of incoming messages should be verified to prevent data injection from other potentially malicious sources.

Layout Considerations As an iframe cannot be sized to it's content's requirements per se, and the consolidated checkout changes it's rendered height depending on the number of available shipping options and selected categories, height changes will trigger an event "pb-on-height-changed" that can be used to change the height of the iframe element in the parent window. The event payload is the height of the unified shipping module in pixels without any margins, so be sure to add any present margins to the value before using it as the iframe size.

// in event handler function
case 'pb-on-height-changed': {
  // change iframe height to render height from iframe content + margins
  document.getElementById('checkoutIFrame').style.height = (value + 30) + 'px';
  break;
}
// ...

Full example page code - for reference

<script>
  // url is replaced with the host url in the documentation template system
  var hostUrl = 'https://widget.porterbuddy-test.com';

  // in a real-world scenario, porterbuddy delivery windows have to be fetched from
  // the porterbuddy availability api
  var deliveryWindows = [
    {
      product: 'delivery',
      start: '2019-03-14T17:30:00+01:00',
      end: '2019-03-14T19:30:00+01:00',
      expiresAt: '2019-03-14T14:30:00+01:00',
      price: {
        fractionalDenomination: 14900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-14T19:30:00+01:00',
      end: '2019-03-14T21:30:00+01:00',
      expiresAt: '2019-03-14T14:30:00+01:00',
      price: {
        fractionalDenomination: 14900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-14T17:30:00+01:00',
      end: '2019-03-14T21:30:00+01:00',
      expiresAt: '2019-03-14T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-15T17:30:00+01:00',
      end: '2019-03-15T19:30:00+01:00',
      expiresAt: '2019-03-15T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-15T19:30:00+01:00',
      end: '2019-03-15T21:30:00+01:00',
      expiresAt: '2019-03-15T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-15T17:30:00+01:00',
      end: '2019-03-15T21:30:00+01:00',
      expiresAt: '2019-03-15T14:30:00+01:00',
      price: {
        fractionalDenomination: 12900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-18T17:30:00+01:00',
      end: '2019-03-18T19:30:00+01:00',
      expiresAt: '2019-03-18T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-18T19:30:00+01:00',
      end: '2019-03-18T21:30:00+01:00',
      expiresAt: '2019-03-18T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-18T17:30:00+01:00',
      end: '2019-03-18T21:30:00+01:00',
      expiresAt: '2019-03-18T14:30:00+01:00',
      price: {
        fractionalDenomination: 12900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-19T17:30:00+01:00',
      end: '2019-03-19T19:30:00+01:00',
      expiresAt: '2019-03-19T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-19T19:30:00+01:00',
      end: '2019-03-19T21:30:00+01:00',
      expiresAt: '2019-03-19T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-19T17:30:00+01:00',
      end: '2019-03-19T21:30:00+01:00',
      expiresAt: '2019-03-19T14:30:00+01:00',
      price: {
        fractionalDenomination: 12900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-20T17:30:00+01:00',
      end: '2019-03-20T19:30:00+01:00',
      expiresAt: '2019-03-20T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-20T19:30:00+01:00',
      end: '2019-03-20T21:30:00+01:00',
      expiresAt: '2019-03-20T14:30:00+01:00',
      price: {
        fractionalDenomination: 13900,
        currency: 'NOK',
      },
    },
    {
      product: 'delivery',
      start: '2019-03-20T17:30:00+01:00',
      end: '2019-03-20T21:30:00+01:00',
      expiresAt: '2019-03-20T14:30:00+01:00',
      price: {
        fractionalDenomination: 12900,
        currency: 'NOK',
      },
    },
  ];

  // delivery options should be fetched from the offered shipping services
  var homeOptions = [
    {
      id: 'porterbuddy',
      name: 'Levert hjem',
      deliveryWindows: deliveryWindows,
      discount: 5000,
      default: true,
    },
    {
      id: 'postnord',
      name: 'Levert hjemme',
      price: {
        fractionalDenomination: 9900,
        currency: 'NOK',
      },
      minDeliveryDays: 2,
      maxDeliveryDays: 5,
      logoUrl: hostUrl + '/logos/postnord.png',
      additionalData: {
        product: 'Standard Insured Parcel',
      }
    },
    {
      id: 'postenhome',
      name: 'Posten på døren',
      logoUrl: hostUrl + '/logos/posten.png',
      levels: [
        {
          id: 'bedrift',
          name: 'Posten Bedriftspakke',
          minDeliveryDays: 1,
          maxDeliveryDays: 3,
          price: {
            fractionalDenomination: 29900,
            currency: 'NOK',
          },
          description: 'Direkte til din bedrift med sporing',
        },
        {
          id: 'express',
          name: 'Posten Expresspakke',
          minDeliveryDays: 1,
          price: {
            fractionalDenomination: 49900,
            currency: 'NOK',
          },
          description: 'Levering innen kl. 09:00 eller kl. 11:30 neste morgen, mandag til fredag'
        }
      ]
    }
  ];

  var pickupOptions = [
    {
      id: 'posten',
      name: 'Hente på utleveringssted',
      price: {
        fractionalDenomination: 5900,
        currency: 'NOK',
      },
      minDeliveryDays: 1,
      maxDeliveryDays: 3,
      locations: [
        {
          id: 'location_1',
          name: 'Coop Mega Sjølyst',
          address: 'Karenslyst Allé 58, 0277 Oslo',
          openingHours: 'Man - Fre: 08:00 - 22:00, Lør: 08:00 - 20:00',
          logoUrl: hostUrl + '/logos/posten.png',
          description: 'Coop Mega er supermarkedet med stort utvalg',
        },
        {
          id: 'location_2',
          name: 'Hoff Post i Butikk',
          address: 'Hoffsveien 10 E, 0275 Oslo',
          openingHours: 'Man - Lør: 06:00 - 23:59',
          logoUrl: hostUrl + '/logos/posten.png',
        },
        {
          id: 'location_3',
          name: 'Elisenberg postkontor',
          address: 'Balchens Gate 7, 0265 Oslo',
          openingHours: 'Man - Fre: 09:00 - 18:00, Lør: 10:00 - 15:00',
          logoUrl: hostUrl + '/logos/postnord.png',
        },
      ],
    },
  ];

  var storeOptions = [
    {
      id: 'pickup',
      name: 'Store pickup',
      minDeliveryDays: 0,
      locations: [
        {
          id: 'shop',
          name: 'Butikken på Skøyen',
          address: 'Karenslyst Allé 16, 0278 Oslo',
          openingHours: 'Man - Fre: 09:00 - 17:00, Lør: stengt',
        },
        {
          id: 'shop2',
          name: 'Butikken på Drammen',
          address: 'Tollbugata 4, 3040 Drammen',
          openingHours: 'Man - Fre: 09:00 - 17:00, Lør: stengt',
        },
      ],
      logoUrl: hostUrl + '/img/test/logo_testshop.png',
    },
  ];

  // the configuration object for the consolidated checkout
  var options = {
    homeDeliveryOptions: [],
    pickupPointOptions: [],
    storeOptions: [],
    resetContext: true,
    showPostCodeInput: true,
    language: 'NO',
    now: '2019-03-14T12:30:00+01:00',
    monitoredProperties: [
      {
        optionId: 'porterbuddy',
        propertyPath: 'data.leaveAtDoorstep',
      },
      {
        optionId: 'porterbuddy',
        propertyPath: 'data.comment',
      }
    ]
  };

  function handlePropertyChange(eventPayload) {
    if (eventPayload.optionId === 'porterbuddy') {
      switch(eventPayload.propertyPath) {
        case 'data.leaveAtDoorstep': {
          document.getElementById("leaveAtDoorStepData").value = eventPayload.value;
          break;
        }
        case 'data.comment': {
          document.getElementById("commentData").value = eventPayload.value;
          break;
        }
        default: {}
      }
    }
  }

  var iframeWindow = null;
  window.addEventListener("message", function(event) {
    if (event.origin === 'https://widget.porterbuddy-test.com' && event.data && event.data.action) {
      let value = event.data.payload;
      switch(event.data.action) {
        case 'pb-frame-ready': {
          iframeWindow = document.getElementById("checkoutIFrame").contentWindow;
          iframeWindow.postMessage({
            action: "pb-checkout-init",
            payload: options
          }, "https://widget.porterbuddy-test.com");
          break;
        }
        case 'pb-on-selection-changed': {
          console.log('Selection changed');
          document.getElementById("selectedType")
            .value = value.category;
          document.getElementById("selectedShippingData")
            .value = JSON.stringify(value.selectedShipping, null, 2);
          break;
        }
        case 'pb-on-selected-property-changed': {
          handlePropertyChange(value);
          break;
        }
        case 'pb-on-postcode-entered': {
          document.getElementById('postCodeData').value = value;
          iframeWindow.postMessage({
            action: 'pb-update-shipping-options',
            payload: {
              homeDeliveryOptions: homeOptions,
              pickupPointOptions: pickupOptions,
              storeOptions: storeOptions,
            }
          }, 'https://widget.porterbuddy-test.com');
          break;
        }
        case 'pb-on-height-changed': {
          document.getElementById('checkoutIFrame').style.height = (value + 30) + 'px';
          break;
        }
        default: {
          console.log('Unknown event ' + JSON.stringify(event.data));
        }
      }
    }
  });

  function refreshCheckout() {
    iframeWindow.postMessage({
      action: 'pb-force-refresh',
      payload: options,
    }, 'https://widget.porterbuddy-test.com');
  }

  function setUserData() {
    var postCode = document.getElementById('postCodeInput').value;
    var email = document.getElementById('emailInput').value;
    iframeWindow.postMessage({
      action: 'pb-set-recipient-info',
      payload: {
        postCode: postCode,
        email: email
      }
    }, 'https://widget.porterbuddy-test.com');
  }

</script>