IMA SDK'ları, multimedya reklamları web sitelerinize ve uygulamalarınıza entegre etmeyi kolaylaştırır. IMA SDK'ları, herhangi bir VAST uyumlu reklam sunucusundan reklam isteğinde bulunabilir ve uygulamalarınızda reklam oynatmayı yönetebilir. IMA DAI SDK'ları ile uygulamalar, reklam ve içerik videosu için (VOD veya canlı içerik) bir akış isteğinde bulunur. SDK daha sonra birleşik bir video akışı döndürür. Böylece uygulamanızda reklam ve içerik videosu arasında geçiş yapmayı yönetmeniz gerekmez.
İlgilendiğiniz DAI çözümünü seçin
Bu kılavuzda, Roku için IMA DAI SDK'yı kullanarak bir DAI Kapsül Yayınlama canlı veya VOD akışının nasıl oynatılacağı gösterilmektedir. Tamamlanmış bir örnek entegrasyonu görüntülemek veya takip etmek için kapsül yayınlama örneğini indirin.
IMA DAI Kapsül Yayınlamaya Genel Bakış
IMA DAI ile kapsül yayınlama özelliğini uygulamak için iki ana SDK bileşeni gerekir. Bu bileşenler bu kılavuzda gösterilmektedir:
- StreamRequest.createPodLiveStreamRequest()/- StreamRequest.createPodVodStreamRequest(): Google'ın reklam sunucularına yönelik bir yayın isteğini tanımlayan bir nesne oluşturur. Bu isteklerde bir Ağ Kodu belirtilir. Pod Live- ima.StreamRequestiçin Özel Öğenin Anahtarı ve isteğe bağlı bir API anahtarı da gerekir.
- StreamManager: İzleme ping'lerini tetikleme ve yayıncıya akış etkinliklerini yönlendirme gibi video akışı ile IMA DAI SDK arasındaki iletişimi yöneten bir nesne.
Ayrıca, uygulamanızın göstermesi için akış manifestini almak üzere manifest manipülasyonu sunucunuza istekte bulunmanız gerekir. Tam süreç, video teknolojisi iş ortağına (VTP) göre değişiklik gösterebilir.
Ön koşullar
- Kullanım alanınızın desteklendiğinden emin olmak için uyumluluk sayfamızı inceleyin.
- Roku örnek oynatıcı kodumuzu indirin.
- Geliştirme kurulumunuzun çalıştığını doğrulamak için örnek oynatıcı kodunu bir Roku cihazına dağıtın.
Videonuzu oynatma
Sağlanan örnek video oynatıcı, ek ayar gerektirmeden içerik videosu oynatır. Geliştirme ortamınızın doğru şekilde ayarlandığından emin olmak için örnek oynatıcıyı Roku cihazınıza dağıtın.
Video oynatıcınızı IMA DAI akış oynatıcısına dönüştürme
Bir akış oynatıcı uygulamak için aşağıdaki adımları uygulayın.
Sdk.xml dosyasını oluşturma
Projenize MainScene.xml ile birlikte Sdk.xml adlı yeni bir dosya ekleyin ve aşağıdaki hazır kodu ekleyin:
Sdk.xml
<?xml version = "1.0" encoding = "utf-8" ?>
  <component name = "imasdk" extends = "Task">
  <interface>
  </interface>
  <script type = "text/brightscript">
  <![CDATA[
    ' Your code goes here.
  ]]>
  </script>
  </component>
Bu kılavuzda MainScene.xml ve Sdk.xml dosyalarının ikisini de düzenlemeniz gerekir.
IMA DAI SDK Çerçevesi'ni yükleme
Çerçeveyi yüklemek için manifest ve Sdk.xml dosyalarına aşağıdakileri ekleyin:
manifest
bs_libs_required=googleima3
Sdk.xml
<?xml version = "1.0" encoding = "utf-8" ?>
  <component name = "imasdk" extends = "Task">
  <interface>
  </interface>
  <script type = "text/brightscript">
  <![CDATA[
  Library "IMA3.brs"
  ]]>
  </script>
  </component>
IMA DAI SDK'sını başlatma
IMA Dinamik Reklam Ekleme akışınızı yüklemenin ilk adımı, IMA DAI SDK'sını yükleyip başlatmaktır. Aşağıdaki kod, IMA DAI SDK komut dosyasını başlatır.
Sdk.xml
<?xml version="1.0" encoding="utf-8" ?>
  <component name="IMASDKTask" extends="Task">
  <interface>
    <field id="IMASDKInitialized" type="Boolean" />
    <field id="errors" type="stringarray" />
  </interface>
  <script type = "text/brightscript">
  <![CDATA[
    Library "IMA3.brs"
    sub init()
      m.top.functionName = "runThread"
    end sub
    sub runThread()
      if not m.top.IMASDKInitialized
        initializeIMASDK()
      end if
    end sub
    sub initializeIMASDK()
        if m.sdk = invalid
          m.sdk = New_IMASDK()
        end if
        m.top.IMASDKInitialized = true
    end sub
  ]]>
  </script>
  </component>
Şimdi bu görevi MainScene.xml içinde başlatın ve içerik akışını yükleme çağrısını kaldırın.
MainScene.xml
<?xml version="1.0" encoding="utf-8" ?>
<component extends="Scene" initialFocus="myVideo" name="MainScene">
  <script type="text/brightscript">
  <![CDATA[
  function init()
    m.video = m.top.findNode("myVideo")
    m.video.notificationinterval = 1
    runIMASDKTask()
  end function
  function runIMASDKTask()
    m.IMASDKTask = createObject("roSGNode", "IMASDKTask")
    m.IMASDKTask.observeField("IMASDKInitialized", "handleIMASDKInitialized")
    m.IMASDKTask.observeField("errors", "handleIMASDKErrors")
    m.IMASDKTask.control = "RUN"
  end function
  sub handleIMASDKInitialized()
    ' Follow your manifest manipulator (VTP) documentation to register a user
    ' streaming session if needed.
  end sub
  sub handleIMASDKErrors(message as object)
    print "------ IMA DAI SDK failed  ------"
    if message <> invalid and message.getData() <> invalid
      print "IMA DAI SDK Error ";message.getData()
    end if
  end sub
  ]]>
  </script>
  <children>
    <Video height="720" id="myVideo" visible="false" width="1280"/>
  </children>
</component>
IMA akış oynatıcı oluşturma
Ardından, mevcut roVideoScreen kullanarak bir IMA akış oynatıcı oluşturmanız gerekir.
Canlı yayın pod'u sunma
Canlı yayınlar için bu akış oynatıcı üç geri çağırma yöntemi uygular: streamInitialized, adBreakStarted ve adBreakEnded.
Akış yüklendiğinde de hızlı oynatma özelliğini devre dışı bırakın. Bu sayede kullanıcıların, reklam arası başladı etkinliği tetiklenmeden önce, videodan önce gösterilen reklamı başladığı anda atlaması engellenir.
Sdk.xml
<?xml version="1.0" encoding="utf-8" ?>
  <component name="IMASDKTask" extends="Task">
  <interface>
    <field id="IMASDKInitialized" type="Boolean" />
    <field id="errors" type="stringarray" />
    <field id="urlData" type="assocarray" />
    <field id="adPlaying" type="Boolean" />
    <field id="videoNode" type="Node" />
  </interface>
  <script type="text/brightscript">
  ...
  sub runThread()
    if not m.top.IMASDKInitialized
      initializeIMASDK()
    end if
    setupPlayerCallbacks()
  end sub
  ...
  sub initializeIMASDK()
    if m.ima = invalid
      ima = New_IMASDK()
      ima.initSdk()
      m.ima = ima
    end if
    m.top.IMASDKInitialized = true
  end sub
  sub setupPlayerCallbacks()
    m.player = m.ima.createPlayer()
    m.player.top = m.top
    m.player.streamInitialized = function(urlData)
      m.top.videoNode.enableTrickPlay = false
      m.top.urlData = urlData
    end function
    m.player.adBreakStarted = function(adBreakInfo)
      print "------ Ad break started ------"
      m.top.adPlaying = true
      m.top.videoNode.enableTrickPlay = false
    end function
    m.player.adBreakEnded = function(adBreakInfo)
      print "------ Ad break ended ------"
      m.top.adPlaying = false
      m.top.videoNode.enableTrickPlay = true
    end function
  end sub
  </script>
...
</component>
VOD akışı kapsülü sunma
VOD akışları için bu akış oynatıcı dört geri çağırma yöntemi uygular: streamInitialized, loadUrl, adBreakStarted ve adBreakEnded. streamInitialized geri aramasında StreamManager.loadThirdPartyStream() numaralı telefonu aradığınızdan emin olun. Aksi takdirde SDK, loadUrl işlevini tetiklemez.
Bu adımda, video teknolojisi iş ortağınızdan (VTP) loadAdPodStream() bölümünde elde edilen yayın kimliğiyle bir yayın URL'si de istersiniz. Ardından, reklam grubu manifestosu ve VTP'niz tarafından döndürülen tüm altyazılarla StreamManager.loadThirdPartyStream() çağrısı yapın.
Akış yüklendiğinde de hızlı oynatma özelliğini devre dışı bırakın. Bu sayede kullanıcıların, reklam arası başladı etkinliği tetiklenmeden önce, videodan önce gösterilen reklamı başladığı anda atlaması engellenir.
Sdk.xml
<?xml version="1.0" encoding="utf-8" ?>
  <component name="IMASDKTask" extends="Task">
  <interface>
    <field id="IMASDKInitialized" type="Boolean" />
    <field id="errors" type="stringarray" />
    <field id="adStitchedStreamInfo" type="assocarray" />
    <field id="adPlaying" type="Boolean" />
    <field id="videoNode" type="Node" />
    <field id="streamParameters" type="assocarray" />
  </interface>
  <script type="text/brightscript">
  ...
  sub runThread()
    if not m.top.IMASDKInitialized
      initializeIMASDK()
    end if
    setupPlayerCallbacks()
  end sub
  ...
  sub initializeIMASDK()
    if m.ima = invalid
      ima = New_IMASDK()
      ima.initSdk()
      m.ima = ima
    end if
    m.top.IMASDKInitialized = true
  end sub
  sub loadThirdPartyStream(adStitchedManifest as string, subtitleConfig as dynamic)
    m.streamManager.loadThirdPartyStream(adStitchedManifest, subtitleConfig)
  end sub
  sub setupPlayerCallbacks()
    m.player = m.ima.createPlayer()
    m.player.top = m.top
    m.player.streamInitialized = function(urlData)
      adStitchedManifest = m.top.streamParameters.VTPManifest.replace("[[STREAMID]]", urlData.streamId)
      loadThirdPartyStream(adStitchedManifest, m.top.streamParameters.subtitleConfig)
    end function
    m.player.loadUrl = function(streamInfo)
      m.top.adStitchedStreamInfo = streamInfo
    end function
    m.player.adBreakStarted = function(adBreakInfo)
      print "------ Ad break started ------"
      m.top.adPlaying = true
      m.top.videoNode.enableTrickPlay = false
    end function
    m.player.adBreakEnded = function(adBreakInfo)
      print "------ Ad break ended ------"
      m.top.adPlaying = false
      m.top.videoNode.enableTrickPlay = true
    end function
  end sub
  </script>
...
</component>
Canlı veya VOD pod yayınlama akışı isteği oluşturma ve yürütme
Bir akış oynatıcınız olduğunda akış isteği oluşturup yürütebilirsiniz.
Bu örnekte, m.testPodServingStream içinde depolanan bir kapsül sunma akışına ait veriler yer alıyor.
Canlı yayın pod'u sunma
m.testPodServingStream nesnesinde, Google Ad Manager'ın söz konusu akışı tanımlamak için ihtiyaç duyduğu parametreleri (ör. ağ kodu ve özel öğe anahtarı) saklayın. Ayrıca, manifest manipülasyon sunucunuza erişmek için kullanılan manifest URL'sini de saklayın. Bu durumda, akış isteği döndürüldükten sonra manifest URL'sine Google akış kimliğinin eklenmesi gerekir.
Reklam Seçenekleri simgeleri gibi AdUI'yi destekleyebilmek için isteğinizin bir parçası olarak içerik videonuzu içeren düğüme bir referans da iletmeniz gerekir.
MainScene.xml
function init()
  m.video = m.top.findNode("myVideo")
  m.video.notificationinterval = 1
  m.testPodServingStream = {
    title: "Test live stream for DAI Pod Serving",
    assetKey: "test-live-stream",
    networkCode: "your-network-code",
    manifest: "https://.../master.m3u8?stream_id=[[STREAMID]]",
    apiKey: ""
  }
  runIMASDKTask()
end function
function runIMASDKTask()
  m.IMASDKTask = createObject("roSGNode", "IMASDKTask")
  m.IMASDKTask.streamParameters = m.testPodservingStream
  m.IMASDKTask.videoNode = m.video
  m.IMASDKTask.observeField("IMASDKInitialized", "handleIMASDKInitialized")
  m.IMASDKTask.observeField("errors", "handleIMASDKErrors")
  m.IMASDKTask.control = "RUN"
end function
Sdk.xml
<interface>
  <field id="IMASDKInitialized" type="Boolean" />
  <field id="errors" type="stringarray" />
  <field id="urlData" type="assocarray" />
  <field id="adPlaying" type="Boolean" />
  <field id="videoNode" type="Node" />
  <field id="streamParameters" type="assocarray" />
</interface>
...
sub runThread()
  if not m.top.IMASDKInitialized
    initializeIMASDK()
  end if
  setupPlayerCallbacks()
  loadAdPodStream()
end sub
sub loadAdPodStream()
  request = m.ima.CreatePodLiveStreamRequest(m.top.streamParameters.assetKey, m.top.streamParameters.networkCode, m.top.streamParameters.apiKey)
  ' Set the player object so that the request can trigger the player's
  ' callbacks at stream initialization or playback events.
  request.player = m.player
  ' Set the video node for the IMA DAI SDK to create ad UI as its child nodes.
  request.adUiNode = m.top.video
  requestResult = m.ima.requestStream(request)
  if requestResult <> invalid
    print "Error requesting stream ";requestResult
    return
  end if
  m.streamManager = invalid
  while m.streamManager = invalid
    sleep(50)
    m.streamManager = m.ima.getStreamManager()
  end while
  if m.streamManager = invalid
    errors = CreateObject("roArray", 1, True)
    invalidStreamManagerError = "Invalid stream manager"
    print invalidStreamManagerError
    errors.push(invalidStreamManagerError)
    m.top.errors = errors
    return
  end if
  if m.streamManager["type"] <> invalid and m.streamManager["type"] = "error"
    errors = CreateObject("roArray", 1, True)
    print "Stream request returns an error. " ; m.streamManager["info"]
    errors.push(m.streamManager["info"])
    m.top.errors = errors
    return
  end if
  setupStreamManager()
  m.streamManager.start()
end sub
VOD akışı kapsülü sunma
m.testPodServingStream nesnesinde, akış isteğinde kullanılan ağ kodunu saklarsınız. Böylece Google Ad Manager bir akış kimliği sağlayabilir. Ayrıca, kullanıcıya özel manifeste erişmek için kullanılan manifest URL'sini manifest işleme sunucunuzda saklayın.
Reklam Seçenekleri simgeleri gibi AdUI'yi destekleyebilmek için isteğinizin bir parçası olarak içerik videonuzu içeren düğüme bir referans da iletmeniz gerekir.
MainScene.xml
sub init()
  m.video = m.top.findNode("myVideo")
  m.video.notificationinterval = 1
  m.testPodServingStream = {
    title: "Pod Serving VOD Stream",
    networkCode: "your-network-code",
    VTPManifest: "https://.../manifest.m3u8?gam-stream-id=[[STREAMID]]",
    subtitleConfig: []
  }
  runIMASDKTask()
end sub
sub runIMASDKTask()
  m.IMASDKTask = createObject("roSGNode", "IMASDKTask")
  m.IMASDKTask.streamParameters = m.testPodservingStream
  m.IMASDKTask.videoNode = m.video
  m.IMASDKTask.observeField("IMASDKInitialized", "handleIMASDKInitialized")
  m.IMASDKTask.observeField("errors", "handleIMASDKErrors")
  m.IMASDKTask.control = "RUN"
end sub
Sdk.xml
sub runThread()
  if not m.top.IMASDKInitialized
    initializeIMASDK()
  end if
  setupPlayerCallbacks()
  loadAdPodStream()
end sub
sub loadAdPodStream()
  request = m.ima.CreatePodVodStreamRequest(m.top.streamParameters.networkCode)
  ' Set the player object so that the request can trigger the player
  ' callbacks at stream initialization or playback events.
  request.player = m.player
  ' Set the video node for the IMA DAI SDK to create ad UI as its child nodes.
  request.adUiNode = m.top.video
  requestResult = m.ima.requestStream(request)
  if requestResult <> invalid
    print "Error requesting stream ";requestResult
    return
  end if
  m.streamManager = invalid
  while m.streamManager = invalid
    sleep(50)
    m.streamManager = m.ima.getStreamManager()
  end while
  if m.streamManager = invalid
    errors = CreateObject("roArray", 1, True)
    invalidStreamManagerError = "Invalid stream manager"
    print invalidStreamManagerError
    errors.push(invalidStreamManagerError)
    m.top.errors = errors
    return
  end if
  if m.streamManager["type"] <> invalid and m.streamManager["type"] = "error"
    errors = CreateObject("roArray", 1, True)
    print "Stream request returns an error. " ; m.streamManager["info"]
    errors.push(m.streamManager["info"])
    m.top.errors = errors
    return
  end if
  setupStreamManager()
  m.streamManager.start()
end sub
Etkinlik işleyicileri ekleme ve yayını başlatma
Canlı yayın pod'u sunma
Yayınınızı istedikten sonra yapmanız gereken birkaç şey kalır: reklam ilerleme durumunu izlemek için etkinlik dinleyicileri ekleyin ve Roku mesajlarını SDK'ya iletin. Doğru reklam oynatma için tüm mesajları SDK'ya yönlendirmeniz önemlidir. Aksi takdirde reklam görüntülemeleri yanlış şekilde raporlanır.
Bu adımda, [[STREAMID]] makrosunu akış kimliğiyle değiştirecek ve tamamlanan manifest isteği URL'sini video oynatıcıya iletecek bir işlev de eklersiniz.
Bu uygulama, bu adımda yayın kimliğini alır ancak VTP entegrasyonunuza bağlı olarak bu adım öncesinde de kullanılabilir.
MainScene.xml
function runIMASDKTask()
  m.IMASDKTask = createObject("roSGNode", "IMASDKTask")
  m.IMASDKTask.streamParameters = m.testPodservingStream
  m.IMASDKTask.videoNode = m.video
  m.IMASDKTask.observeField("IMASDKInitialized", "handleIMASDKInitialized")
  m.IMASDKTask.observeField("errors", "handleIMASDKErrors")
  m.sdkTask.observeField("adStitchedStreamInfo", "loadAdStitchedStream")
  m.sdkTask.control = "RUN"
end function
sub loadAdStitchedStream(message as object)
  print "Ad pod stream information ";message
  adPodStreamInfo = message.getData()
  manifest = m.testPodservingStream.manifest.Replace("[[STREAMID]]", adPodStreamInfo.streamId)
  playStream(manifest, adPodStreamInfo.format)
end sub
sub playStream(url as string, format as string)
  vidContent = createObject("RoSGNode", "ContentNode")
  vidContent.url = url
  vidContent.title = m.testPodservingStream.title
  vidContent.streamformat = format
  m.video.content = vidContent
  m.video.setFocus(true)
  m.video.visible = true
  m.video.control = "play"
  m.video.EnableCookies()
end sub
Sdk.xml
sub runThread()
  if not m.top.IMASDKInitialized
    initializeIMASDK()
  end if
  setupPlayerCallbacks()
  loadAdPodStream()
  if m.streamManager <> invalid
    runLoop()
  end if
end sub
sub runLoop()
  m.top.videoNode.timedMetaDataSelectionKeys = ["*"]
  ' IMPORTANT: Failure to listen to the position and timedmetadata fields
  ' could result in ad impressions not being reported.
  m.port = CreateObject("roMessagePort")
  m.top.videoNode.observeField("position", m.port)
  m.top.videoNode.observeField("timedMetaData", m.port)
  m.top.videoNode.observeField("timedMetaData2", m.port)
  m.top.videoNode.observeField("state", m.port)
  while True
    msg = wait(1000, m.port)
    if m.top.videoNode = invalid
      print "exiting"
      exit while
    end if
    m.streamManager.onMessage(msg)
    currentTime = m.top.videoNode.position
    if currentTime > 3 And not m.top.adPlaying
      m.top.videoNode.enableTrickPlay = true
    end if
  end while
end sub
sub setupStreamManager()
  m.streamManager.addEventListener(m.sdk.AdEvent.ERROR, errorCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.START, startCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.FIRST_QUARTILE, firstQuartileCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.MIDPOINT, midpointCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.THIRD_QUARTILE, thirdQuartileCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.COMPLETE, completeCallback)
end sub
sub startCallback(ad as object)
  print "Callback from SDK -- Start called - "
end sub
sub firstQuartileCallback(ad as object)
  print "Callback from SDK -- First quartile called - "
end sub
sub midpointCallback(ad as object)
  print "Callback from SDK -- Midpoint called - "
end sub
sub thirdQuartileCallback(ad as object)
  print "Callback from SDK -- Third quartile called - "
end sub
sub completeCallback(ad as object)
  print "Callback from SDK -- Complete called - "
end sub
function errorCallback(error as object)
  print "Callback from SDK -- Error called - " ; error
  m.errorState = True
end function
VOD akışı kapsülü sunma
Yayınınızı istedikten sonra yapmanız gereken birkaç şey kalır: reklam ilerleme durumunu izlemek için etkinlik dinleyicileri ekleyin ve Roku mesajlarını SDK'ya iletin. Doğru reklam oynatma için tüm iletileri SDK'ya yönlendirmeniz önemlidir. Aksi takdirde reklam görüntülemeleri yanlış raporlanır.
MainScene.xml
sub runIMASDKTask()
  m.IMASDKTask = createObject("roSGNode", "IMASDKTask")
  m.IMASDKTask.streamParameters = m.testPodservingStream
  m.IMASDKTask.videoNode = m.video
  m.IMASDKTask.observeField("IMASDKInitialized", "handleIMASDKInitialized")
  m.IMASDKTask.observeField("errors", "handleIMASDKErrors")
  m.sdkTask.observeField("adStitchedStreamInfo", "loadAdStitchedStream")
  m.sdkTask.control = "RUN"
end sub
sub loadAdStitchedStream(message as object)
  print "Ad pod stream information ";message
  adPodStreamInfo = message.getData()
end sub
sub playStream(url as string, format as string, subtitleConfig as object)
  vidContent = createObject("RoSGNode", "ContentNode")
  vidContent.title = m.testPodservingStream.title
  vidContent.url = url
  vidContent.subtitleConfig = subtitleConfig
  vidContent.streamformat = format
  m.video.content = vidContent
  m.video.setFocus(true)
  m.video.visible = true
  m.video.control = "play"
  m.video.EnableCookies()
end sub
Sdk.xml
sub runThread()
  if not m.top.IMASDKInitialized
    initializeIMASDK()
  end if
  setupPlayerCallbacks()
  loadAdPodStream()
  if m.streamManager <> invalid
    runLoop()
  end if
end sub
sub runLoop()
  m.top.videoNode.timedMetaDataSelectionKeys = ["*"]
  ' IMPORTANT: Failure to listen to the position and timedmetadata fields
  ' could result in ad impressions not being reported.
  m.port = CreateObject("roMessagePort")
  m.top.videoNode.observeField("position", m.port)
  m.top.videoNode.observeField("timedMetaData", m.port)
  m.top.videoNode.observeField("timedMetaData2", m.port)
  m.top.videoNode.observeField("state", m.port)
  while True
    msg = wait(1000, m.port)
    if m.top.videoNode = invalid
      exit while
    end if
    m.streamManager.onMessage(msg)
    currentTime = m.top.videoNode.position
    if currentTime > 3 and not m.top.adPlaying
      m.top.videoNode.enableTrickPlay = true
    end if
  end while
end sub
sub setupStreamManager()
  m.streamManager.addEventListener(m.sdk.AdEvent.ERROR, errorCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.START, startCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.FIRST_QUARTILE, firstQuartileCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.MIDPOINT, midpointCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.THIRD_QUARTILE, thirdQuartileCallback)
  m.streamManager.addEventListener(m.sdk.AdEvent.COMPLETE, completeCallback)
end sub
sub startCallback(ad as object)
  print "Callback from SDK -- Start called - "
end sub
sub firstQuartileCallback(ad as object)
  print "Callback from SDK -- First quartile called - "
end sub
sub midpointCallback(ad as object)
  print "Callback from SDK -- Midpoint called - "
end sub
sub thirdQuartileCallback(ad as object)
  print "Callback from SDK -- Third quartile called - "
end sub
sub completeCallback(ad as object)
  print "Callback from SDK -- Complete called - "
end sub
sub errorCallback(error as object)
  print "Callback from SDK -- Error called - " ; error
  m.errorState = True
end sub