맞춤 네이티브 광고 형식

맞춤 네이티브 광고 형식

Ad Manager 게시자는 시스템 정의 네이티브 형식 외에도 맞춤 애셋 목록을 정의하여 자체 네이티브 광고 형식을 만들 수 있습니다. 이를 맞춤 네이티브 광고 형식이라고 하며, 예약 광고와 함께 사용할 수 있습니다. 이를 통해 게시자는 임의의 구조화된 데이터를 앱에 전달할 수 있습니다. 이러한 광고는 NativeCustomFormatAd 객체로 표시됩니다.

맞춤 네이티브 광고 형식 로드

이 가이드에서는 맞춤 네이티브 광고 형식을 로드하고 표시하는 방법을 설명합니다.

PPID 구축

네이티브 광고처럼 맞춤 네이티브 광고 형식은 AdLoader 클래스를 통해 로드됩니다.

Java

AdLoader adLoader = new AdLoader.Builder(context, "/6499/example/native")
    .forCustomFormatAd("10063170",
      new NativeCustomFormatAd.OnCustomFormatAdLoadedListener() {
          @Override
          public void onCustomFormatAdLoaded(NativeCustomFormatAd ad) {
              // Show the custom format and record an impression.
          }
      },
      new NativeCustomFormatAd.OnCustomClickListener() {
          @Override
          public void onCustomClick(NativeCustomFormatAd ad, String s) {
              // Handle the click action
          }
      })
    .withAdListener( ... )
    .withNativeAdOptions( ... )
    .build();

Kotlin

val adLoader = AdLoader.Builder(this, "/6499/example/native")
        .forCustomFormatAd("10063170",
            { ad ->
                // Show the custom format and record an impression.
            },
            { ad, s ->
            // Handle the click action
            })
        .withAdListener( ... )
        .withNativeAdOptions( ... )
        .build()

forCustomFormatAd 메서드는 맞춤 네이티브 광고 형식을 요청하도록 AdLoader를 구성합니다. 이 메서드에는 세 가지 매개변수가 전달됩니다.

  • AdLoader에서 요청해야 하는 맞춤 네이티브 광고 형식의 ID입니다. 각 맞춤 네이티브 광고 형식에는 ID가 연결되어 있습니다. 이 매개변수는 앱에서 AdLoader를 통해 요청할 형식을 나타냅니다.
  • 광고가 성공적으로 로드되면 호출되는 OnCustomFormatAdLoadedListener.
  • 사용자가 광고를 탭하거나 클릭할 때 호출될 OnCustomClickListener(선택사항). 이 리스너에 대한 자세한 내용은 아래의 '클릭 및 노출 처리' 섹션을 참조하세요.

두 개 이상의 광고 소재 형식을 게재하도록 단일 광고 단위를 설정할 수 있으므로, 광고 로더가 2개 이상의 가능한 맞춤 네이티브 광고 형식을 게재하도록 고유한 형식 ID로 forCustomFormatAd를 여러 번 호출할 수 있습니다.

맞춤 네이티브 광고 형식 ID

맞춤 네이티브 광고 형식을 식별하는 데 사용되는 형식 ID는 Ad Manager UI의 네이티브 섹션 내 게재 드롭다운에서 확인할 수 있습니다.

각 맞춤 네이티브 광고 형식 ID가 이름 옆에 표시됩니다. 이름 중 하나를 클릭하면 형식의 필드에 대한 정보를 보여주는 세부정보 화면이 표시됩니다.

여기에서 개별 필드를 추가, 수정, 삭제할 수 있습니다. 각 애셋의 이름을 기록해 둡니다. 이름은 맞춤 네이티브 광고 형식을 표시할 때 각 애셋의 데이터를 가져오는 데 사용되는 키입니다.

맞춤 네이티브 광고 형식 표시

맞춤 네이티브 광고 형식은 게시자가 광고를 구성하는 자체 애셋 목록을 정의할 수 있다는 점에서 시스템 정의 광고 형식과 다릅니다. 따라서 맞춤 네이티브 광고 형식의 표시 방법은 몇 가지 측면에서 시스템 정의 형식과 다릅니다.

  1. NativeCustomFormatAd 클래스는 Ad Manager에서 정의한 맞춤 네이티브 광고 형식을 처리하는 것이기 때문에 애셋에 'getters'라는 이름을 지정하지 않습니다. 대신 필드 이름을 매개변수로 사용하는 getText, getImage 등의 메서드를 제공합니다.
  2. NativeCustomFormatAd와 함께 사용하는 전용 광고 뷰 클래스(예: NativeAdView)는 없습니다. 사용자 환경에 적합한 모든 레이아웃을 자유롭게 사용할 수 있습니다.
  3. 전용 ViewGroup 클래스가 없으므로 광고 애셋을 표시하는 데 사용하는 뷰를 등록하지 않아도 됩니다. 이렇게 하면 광고를 표시할 때 코드 몇 줄이 절약되지만, 나중에 클릭을 처리하기 위해 약간의 작업을 추가로 수행해야 합니다.

다음은 NativeCustomFormatAd를 표시하는 함수의 예입니다.

Java

public void displayCustomFormatAd (ViewGroup parent,
                                     NativeCustomFormatAd customFormatAd) {
    // Inflate a layout and add it to the parent ViewGroup.
    LayoutInflater inflater = (LayoutInflater) parent.getContext()
            .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    View adView = inflater.inflate(R.layout.custom_format_ad, parent);

    // Locate the TextView that will hold the value for "Headline" and
    // set its text.
    TextView myHeadlineView = (TextView) adView.findViewById(R.id.headline);
    myHeadlineView.setText(customFormatAd.getText("Headline"));

    // Locate the ImageView that will hold the value for "MainImage" and
    // set its drawable.
    Button myMainImageView = (ImageView) adView.findViewById(R.id.main_image);
    myMainImageView.setImageDrawable(
            customFormatAd.getImage("MainImage").getDrawable());

    ...
    // Continue locating views and displaying assets until finished.
    ...
}

Kotlin

public fun displayCustomFormatAd (parent: ViewGroup,
                                customFormatAd: NativeCustomFormatAd) {
    val adView = layoutInflater
            .inflate(R.layout.ad_simple_custom_format, null)

    val myHeadlineView = adView.findViewById<TextView>(R.id.headline)
    myHeadlineView.setText(customFormatAd.getText("Headline"));

    // Locate the ImageView that will hold the value for "MainImage" and
    // set its drawable.
    val myMainImageView = adView.findViewById(R.id.main_image);
    myMainImageView.setImageDrawable(
            customFormatAd.getImage("MainImage").drawable;

    ...
    // Continue locating views and displaying assets until finished.
    ...
}

맞춤 네이티브 광고 형식의 네이티브 동영상

맞춤 형식을 만들 때 형식을 동영상에 사용할 수 있도록 만들 수 있습니다.

앱 구현에서 NativeCustomFormatAd.getMediaContent()를 사용하여 미디어 콘텐츠를 가져올 수 있습니다. 그런 다음 setMediaContent()를 호출하여 미디어 뷰의 미디어 콘텐츠를 미디어 뷰로 설정합니다. 광고에 동영상 콘텐츠가 없으면 동영상 없이 광고를 표시하는 대체 계획을 세웁니다.

다음은 광고에 동영상 콘텐츠가 있는지 확인하고 동영상을 사용할 수 없는 경우 대신 이미지를 표시하는 예입니다.

Java

// Called when a custom native ad loads.
@Override
public void onCustomFormatAdLoaded(final NativeCustomFormatAd ad) {

  MediaContent mediaContent = ad.getMediaContent();

  // Assumes you have a FrameLayout in your view hierarchy with the id media_placeholder.
  FrameLayout mediaPlaceholder = (FrameLayout) findViewById(R.id.media_placeholder);

  // Apps can check the MediaContent's hasVideoContent property to determine if the
  // NativeCustomFormatAd has a video asset.
  if (mediaContent != null && mediaContent.hasVideoContent()) {
    MediaView mediaView = new MediaView(mediaPlaceholder.getContext());
    mediaView.setMediaContent(mediaContent);
    mediaPlaceholder.addView(mediaView);

    // Create a new VideoLifecycleCallbacks object and pass it to the VideoController. The
    // VideoController will call methods on this object when events occur in the video
    // lifecycle.
    VideoController vc = mediaContent.getVideoController();
    vc.setVideoLifecycleCallbacks(
        new VideoController.VideoLifecycleCallbacks() {
          @Override
          public void onVideoEnd() {
            // Publishers should allow native ads to complete video playback before
            // refreshing or replacing them with another ad in the same UI location.
            super.onVideoEnd();
          }
        });
  } else {
    ImageView mainImage = new ImageView(this);
    mainImage.setAdjustViewBounds(true);
    mainImage.setImageDrawable(ad.getImage("MainImage").getDrawable());
    mediaPlaceholder.addView(mainImage);
    mainImage.setOnClickListener(
        new View.OnClickListener() {
          @Override
          public void onClick(View view) {
            ad.performClick("MainImage");
          }
        });
  }
}

Kotlin

// Called when a custom native ad loads.
NativeCustomFormatAd.OnCustomFormatAdLoadedListener { ad ->

  val mediaContent = ad.mediaContent

  // Apps can check the MediaContent's hasVideoContent property to determine if the
  // NativeCustomFormatAd has a video asset.
  if (mediaContent != null && mediaContent.hasVideoContent()) {
    val mediaView = MediaView(mediaPlaceholder.getContest())
    mediaView.mediaContent = mediaContent

    val videoController = mediaContent.videoController

    // Create a new VideoLifecycleCallbacks object and pass it to the VideoController. The
    // VideoController will call methods on this object when events occur in the video
    // lifecycle.
    if (videoController != null) {
      videoController.videoLifecycleCallbacks =
        object : VideoController.VideoLifecycleCallbacks() {
          override fun onVideoEnd() {
            // Publishers should allow native ads to complete video playback before refreshing
            // or replacing them with another ad in the same UI location.
            super.onVideoEnd()
          }
        }
    }
  } else {
    val mainImage = ImageView(this)
    mainImage.adjustViewBounds = true
    mainImage.setImageDrawable(ad.getImage("MainImage")?.drawable)

    mainImage.setOnClickListener { ad.performClick("MainImage") }
    customTemplateBinding.simplecustomMediaPlaceholder.addView(mainImage)
  }
}

맞춤 네이티브 광고의 동영상 환경을 맞춤설정하는 방법에 관한 자세한 내용은 MediaContent를 참고하세요.

실제 사용 중인 네이티브 동영상의 작동 예시에 대한 Ad Manager 맞춤 렌더링 예시를 다운로드하세요.

맞춤 네이티브 광고 형식 클릭수 및 노출수

맞춤 네이티브 광고 형식을 사용하면 앱에서 노출을 기록하고 Google 모바일 광고 SDK에 클릭 이벤트를 보고해야 합니다.

노출수 기록

맞춤 형식 광고의 노출을 기록하려면 해당 NativeCustomFormatAd에서 recordImpression 메서드를 호출합니다.

myCustomFormatAd.recordImpression();

앱에서 실수로 동일한 광고에 대해 메서드를 두 번 호출하면 SDK가 자동으로 단일 요청에 대해 중복 노출이 기록되지 않도록 합니다.

클릭수 보고

애셋 보기에서 클릭이 발생했음을 SDK에 보고하려면 해당 NativeCustomFormatAd에서 performClick 메서드를 호출하고 클릭한 애셋의 이름을 전달하세요. 예를 들어 'MainImage'라는 맞춤 형식의 애셋이 있고 이 애셋에 해당하는 ImageView의 클릭을 보고하려는 경우 코드는 다음과 같습니다.

myCustomFormatAd.performClick("MainImage");

광고와 연결된 모든 보기에 대해 이 메서드를 호출할 필요는 없습니다. 'Caption'이라는 다른 필드를 표시해야 하지만 사용자가 클릭하거나 탭하지 않은 경우에는 앱에서 해당 애셋 보기에 대해 performClick를 호출하지 않아도 됩니다.

맞춤 클릭 액션에 응답

맞춤 형식 광고를 클릭하면 SDK에서 세 가지 가능한 응답이 다음 순서대로 시도됩니다.

  1. AdLoader에서 OnCustomClickListener를 호출합니다(제공된 경우).
  2. 광고의 딥 링크 URL별로 콘텐츠 리졸버를 찾고 리졸버하는 첫 번째 리졸버를 시작합니다.
  3. 브라우저를 열고 광고의 기존 도착 URL로 이동합니다.

forCustomFormatAd 메서드는 OnCustomClickListener를 허용합니다. 리스너 객체를 전달하면 SDK가 onCustomClick 메서드를 호출하고 더 이상의 조치를 취하지 않습니다. 그러나 null 값을 리스너로 전달하면 SDK는 광고에 등록된 딥 링크 또는 도착 URL로 대체됩니다.

맞춤 클릭 리스너를 사용하면 UI 업데이트, 새 활동 실행 또는 단순한 클릭 기록 등 클릭에 대한 응답으로 취할 수 있는 최상의 조치를 앱에서 결정할 수 있습니다. 다음은 클릭이 발생했음을 간단히 기록하는 예입니다.

Java

AdLoader adLoader = new AdLoader.Builder(context, "/6499/example/native")
    .forCustomFormatAd("10063170",
      new NativeCustomFormatAd.OnCustomFormatAdLoadedListener() {
        // Display the ad.
      },
      new NativeCustomFormatAd.OnCustomClickListener() {
          @Override
          public void onCustomClick(NativeCustomFormatAd ad, String assetName) {
            Log.i("MyApp", "A custom click just happened for " + assetName + "!");
          }
      }).build();

Kotlin

val adLoader = AdLoader.Builder(this, "/6499/example/native")
    .forCustomFormatAd("10063170",
        { ad ->
            // Display the ad.
        },
        { ad, assetName ->
                Log.i("MyApp", "A custom click just happened for $assetName!")
    }).build()

처음에는 맞춤 클릭 리스너가 있다는 것이 이상하게 보일 수 있습니다. 앱에서 SDK에 클릭이 발생했음을 알렸는데 SDK가 이를 앱에 다시 보고하는 이유는 무엇일까요?

이러한 정보 흐름이 유용한 데에는 몇 가지 이유가 있지만 가장 중요한 점은 SDK가 클릭에 대한 응답을 계속 제어할 수 있다는 것입니다. 예를 들어 광고 소재에 설정된 서드 파티 추적 URL을 자동으로 핑하고, 추가 코드 없이 다른 작업을 백그라운드에서 처리할 수 있습니다.