راهنمای دستگاه دوربین برای اندروید

نوع دستگاه دوربین با استفاده از دو ویژگی پیاده‌سازی می‌شود: PushAvStreamTransport که انتقال جریان صدا و تصویر را با استفاده از پروتکل‌های مبتنی بر فشار مدیریت می‌کند، و WebRtcLiveView که امکان کنترل پخش زنده و talkback را فراهم می‌کند.

قبل از استفاده از هرگونه ویژگی یا تلاش برای به‌روزرسانی ویژگی‌ها، همیشه پشتیبانی از ویژگی‌ها و دستورات را برای دستگاه بررسی کنید. به بخش کنترل دستگاه‌ها مراجعه کنید.Android برای اطلاعات بیشتر.

نوع دستگاه APIهای خانگی صفات نمونه برنامه کاتلین مورد استفاده

دوربین

Google Camera Device

home.matter.6006.types.0158

دستگاهی که تصاویر ثابت یا ویدیو ضبط می‌کند. دوربین‌ها ممکن است دارای قابلیت پخش زنده، مکالمه دو طرفه یا تشخیص رویدادها باشند.

ویژگی‌های مورد نیاز
گوگل PushAvStreamTransport
گوگل وب‌آر‌تی‌سی لایو ویو

دوربین

دریافت اطلاعات اولیه در مورد یک دستگاه

ویژگی BasicInformation شامل اطلاعاتی مانند نام فروشنده، شناسه فروشنده، شناسه محصول، نام محصول (شامل اطلاعات مدل)، نسخه نرم‌افزار و شماره سریال دستگاه است:

// Get device basic information. All general information traits are on the RootNodeDevice type.
    device.type(RootNodeDevice).first().standardTraits.basicInformation?.let { basicInformation ->
        println("vendorName ${basicInformation.vendorName}")
        println("vendorId ${basicInformation.vendorId}")
        println("productId ${basicInformation.productId}")
        println("productName ${basicInformation.productName}")
        println("softwareVersion ${basicInformation.softwareVersion}")
        println("serialNumber ${basicInformation.serialNumber}")
    }

آخرین زمان تماس ابری دستگاه را دریافت کنید

برای یافتن آخرین زمانی که دستگاه با فضای ابری تماس داشته است، از ویژگی lastContactTimestamp از ویژگی ExtendedGeneralDiagnostics استفاده کنید:

fun getLastContactTimeStamp(trait: ExtendedGeneralDiagnostics): java.time.Instant {
  val timestamp = trait.lastContactTimestamp
  return Instant.ofEpochSecond(timestamp.toLong())
}

بررسی اتصال دستگاه

اتصال برای یک دستگاه در واقع در سطح نوع دستگاه بررسی می‌شود زیرا برخی از دستگاه‌ها از چندین نوع دستگاه پشتیبانی می‌کنند. حالت برگردانده شده ترکیبی از حالت‌های اتصال برای همه ویژگی‌های آن دستگاه است.

val lightConnectivity = dimmableLightDevice.metadata.sourceConnectivity.connectivityState

در صورت عدم اتصال به اینترنت، ممکن است در مورد انواع دستگاه‌های مختلف، حالت PARTIALLY_ONLINE مشاهده شود. ویژگی‌های استاندارد Matter ممکن است به دلیل مسیریابی محلی هنوز آنلاین باشند، اما ویژگی‌های مبتنی بر ابر آفلاین خواهند بود.

شروع پخش زنده

برای شروع پخش زنده، رشته‌ی پروتکل شرح جلسه (SDP) را به متد startLiveView() از ویژگی WebRtcLiveView ارسال کنید، که یک WebRtcLiveViewTrait.StartLiveViewCommand.Response حاوی سه مقدار را برمی‌گرداند:

  • SDP برای جلسه.
  • مدت زمان جلسه بر حسب ثانیه.
  • شناسه جلسه، که ممکن است برای تمدید یا خاتمه جلسه استفاده شود.
suspend fun getWebRtcLiveViewTrait(cameraDevice: HomeDevice) {
 return cameraDevice.type(GoogleCameraDevice).trait(WebRtcLiveView).first {
    it?.metadata?.sourceConnectivity?.connectivityState == ConnectivityState.ONLINE
  }

}

// Start the live view
suspend fun startCameraStream(trait: WebRtcLiveView, offerSdp: String) {
  val response = trait.startLiveView(offerSdp)
  // Response contains three fields (see below)
  return response
}
  ...

// This is used to manage the WebRTC connection
val peerConnection: RTCPeerConnection = ...

   ...

val startResponse = startCameraStream(sdp)
val answerSdp = startResponse?.answerSdp
val sessionDuration = startResponse?.liveSessionDurationSeconds
val mediaSessionId = startResponse?.mediaSessionId

peerConnection.setRemoteDescription(SessionDescription.Type.ANSWER,
                                    answerSdp)

پخش زنده را تمدید کنید

پخش زنده دارای مدت زمان از پیش تعیین شده‌ای است که پس از آن منقضی می‌شود. برای افزایش مدت زمان یک پخش زنده فعال، با استفاده از متد WebRtcLiveView.extendLiveView() درخواست تمدید ارسال کنید:

// Assuming camera stream has just been started
suspend fun scheduleExtension(trait: WebRtcLiveView, mediaSessionId: String, liveSessionDurationSeconds: UShort ) {
  delay(liveSessionDurationSeconds - BUFFER_SECONDS * 1000)
  val response = trait.extendLiveView(mediaSessionId)
  // returns how long the session will be live for
  return response.liveSessionDurationSeconds
}

شروع و توقف تاک‌بک

برای شروع talkback، متد startTalkback() از ویژگی WebRtcLiveView را فراخوانی کنید. برای توقف آن، stopTalkback() استفاده کنید.

// Make sure camera stream is on
suspend fun setTalkback(isOn: Boolean, trait: WebRtcLiveView, mediaSessionId: String) {
  if(isOn) {
    trait.startTalkback(mediaSessionId)
  } else {
    trait.stopTalkback(mediaSessionId)
  }
}

فعال و غیرفعال کردن قابلیت ضبط

برای فعال کردن قابلیت ضبط دوربین، TransportStatusEnum.Active را به متد setTransportStatus() از ویژگی PushAvStreamTransport ارسال کنید. برای غیرفعال کردن قابلیت ضبط، TransportStatusEnum.Inactive را ارسال کنید. در مثال زیر، این فراخوانی‌ها را در یک فراخوانی واحد قرار می‌دهیم که از یک Boolean برای فعال یا غیرفعال کردن قابلیت ضبط استفاده می‌کند:

// Start or stop recording for all connections.
suspend fun setCameraRecording(trait: PushAvStreamTransport, isOn: Boolean) {
  if(isOn) {
    trait.setTransportStatus(TransportStatusEnum.Active)
  } else {
    trait.setTransportStatus(TransportStatusEnum.Inactive)
  }
}

فعال یا غیرفعال کردن قابلیت ضبط دوربین، همانند روشن یا خاموش کردن فیلم‌برداری دوربین است. وقتی فیلم‌برداری دوربین فعال است، در حال ضبط است (برای رویدادها و کلیپ‌های مرتبط).

وقتی قابلیت ضبط غیرفعال است (فیلمبرداری دوربین خاموش است):

  • دوربین همچنان می‌تواند بر اساس connectivityState نوع دستگاه، آنلاین نشان داده شود.
  • دسترسی به پخش زنده امکان‌پذیر نیست و دوربین نیز هیچ رویداد ابری را تشخیص نمی‌دهد.

بررسی کنید که آیا قابلیت ضبط فعال است یا خیر

برای تعیین اینکه آیا قابلیت ضبط دوربین فعال است یا خیر، بررسی کنید که آیا اتصالات فعال هستند یا خیر. مثال زیر دو تابع برای انجام این کار تعریف می‌کند:

// Get the on/off state
suspend fun onOffState(pushAvStreamTransport: PushAvStreamTransport) {
  return pushAvStreamTransport
    .currentConnections?.any { it.transportStatus == TransportStatusEnum.Active } ?: false
}

// Check if the camera's recording capability is enabled
fun PushAvStreamTransport.recordModeActive(): Boolean {
  return currentConnections?.any { it.transportStatus == TransportStatusEnum.Active } ?: false
}

راه دیگر برای بررسی، استفاده از تابع findTransport() به همراه یک گزاره است:

// Fetch the current connections
suspend fun queryRecordModeState(trait: PushAvStreamTransport) {
  return trait.findTransport().let {
      it.transportConfigurations.any { it.transportStatus == TransportStatusEnum.Active
    }
}

تنظیمات باتری

تنظیمات مختلف باتری را می‌توان از طریق APIهای Home کنترل کرد.

تنظیم اولویت استفاده از باتری

تنظیم تعادل انرژی به شما امکان می‌دهد تا بین عمر باتری و عملکرد دستگاه تعادل برقرار کنید. می‌توانید پروفایل‌های باتری مختلفی مانند «توسعه‌یافته»، «متعادل» و «عملکرد» ایجاد کنید و بین آنها جابجا شوید.

این ویژگی با به‌روزرسانی ویژگی currentEnergyBalance از ویژگی EnergyPreference پیاده‌سازی می‌شود. این ویژگی یک اندیس عدد صحیح می‌پذیرد که مربوط به یک پروفایل خاص تعریف شده در لیست energyBalances دستگاه است (برای مثال، 0 برای EXTENDED ، 1 برای BALANCED و 2 برای PERFORMANCE ).

مقدار null برای currentEnergyBalance نشان می‌دهد که دستگاه از یک پروفایل سفارشی استفاده می‌کند. این حالت فقط خواندنی است.

در ادامه مثالی از ساختاری که ویژگی currentEnergyBalance از آن استفاده خواهد کرد، و به دنبال آن قطعه کد واقعی که از این ویژگی استفاده می‌کند، نشان داده شده است.

// Example energyBalances list
energy_balances: [
  {
    step: 0,
    label: "EXTENDED"
  },
  {
    step: 50,
    label: "BALANCED"
  },
  {
    step: 100,
    label: "PERFORMANCE"
  }
]
// The index parameter must be within the UByte range (0-255).
suspend fun setEnergyBalance(trait: EnergyPreference, index: Int) {
  trait.update { setCurrentEnergyBalance(index.toUByte()) }
}

// Setting the battery usage to more recording ie performance
setEnergyBalance(energyPreference, 2)

فعال کردن ذخیره خودکار باتری

برای پیکربندی این ویژگی، ویژگی currentLowPowerModeSensitivity از ویژگی EnergyPreference را به‌روزرسانی کنید. این ویژگی از یک شاخص برای انتخاب سطح حساسیت استفاده می‌کند، که در آن 0 معمولاً نشان دهنده Disabled و 1 نشان دهنده Enabled یا Automatic ) است.

suspend fun setAutomaticBatterySaver(enable: Boolean, trait: EnergyPreference) {
  // 0 is Disabled, 1 is Enabled
  val value = if (enable) 1.toUByte() else 0.toUByte()
  trait.update { setCurrentLowPowerModeSensitivity(value) }
}

دریافت وضعیت شارژ باتری

برای دریافت وضعیت شارژ فعلی دستگاه (در حال شارژ، کاملاً شارژ شده یا در حال شارژ نیست)، از ویژگی batChargeState از ویژگی PowerSource استفاده کنید.

// Get the battery charging state
val batteryChargeState = powerSource.batChargeState

when (batteryChargeState) {
    PowerSourceTrait.BatChargeStateEnum.IsCharging -> "Charging"
    PowerSourceTrait.BatChargeStateEnum.IsAtFullCharge -> "Full"
    PowerSourceTrait.BatChargeStateEnum.IsNotCharging -> "Not Charging"
    else -> "Unknown"
}

دریافت میزان باتری

برای دریافت سطح فعلی باتری، از ویژگی batChargeLevel از ویژگی PowerSource استفاده کنید. این سطح یا OK ، Warning (کم) یا Critical است.

// Get the battery charge level
val batteryLevel = powerSourceTrait.batChargeLevel

when (batteryLevel) {
    PowerSourceTrait.BatChargeLevelEnum.OK -> "OK"
    PowerSourceTrait.BatChargeLevelEnum.Warning -> "Warning"
    PowerSourceTrait.BatChargeLevelEnum.Critical -> "Critical"
    else -> "Unknown"
}

منبع تغذیه را دریافت کنید

برای تعیین منبع تغذیه‌ای که دستگاه از آن استفاده می‌کند، از ویژگی‌های BatPresent و wiredPresent از ویژگی PowerSource استفاده کنید.

  val trait: PowerSource
  val isWired = trait.wiredPresent
  val hasBattery = trait.batPresent

تنظیمات صوتی

تنظیمات مختلف صوتی را می‌توان از طریق APIهای Home کنترل کرد.

میکروفون را روشن یا خاموش کنید

برای روشن یا خاموش کردن میکروفون دستگاه، ویژگی microphoneMuted از ویژگی CameraAvStreamManagement را با استفاده از تابع داخلی setMicrophoneMuted کاتلین به‌روزرسانی کنید:

// Turn the device's microphone on or off
suspend fun turnOffMicrophone(disableMicrophone: Boolean, trait: CameraAvStreamManagement) {
  trait.update { setMicrophoneMuted(disableMicrophone) }
}

ضبط صدا را روشن یا خاموش کنید

برای فعال یا غیرفعال کردن ضبط صدا برای دستگاه، ویژگی recordingMicrophoneMuted از ویژگی CameraAvStreamManagement را با استفاده از تابع داخلی setRecordingMicrophoneMuted کاتلین به‌روزرسانی کنید:

// Turn audio recording on or off for the device
suspend fun turnOffAudioRecording(disableAudioRecording: Boolean, trait: CameraAvStreamManagement) {
  trait.update { setRecordingMicrophoneMuted(disableAudioRecording) }
}

تنظیم صدای بلندگو

برای تنظیم میزان صدای بلندگو برای دستگاه، ویژگی speakerVolumeLevel از ویژگی CameraAvStreamManagement را با استفاده از تابع داخلی setSpeakerVolumeLevel کاتلین به‌روزرسانی کنید:

// Adjust the camera speaker volume
suspend fun adjustSpeakerVolume(volume: Int, trait: CameraAvStreamManagement) {
  trait.update { setSpeakerVolumeLevel(volume.toUbyte()) }
}

تنظیمات منطقه فعالیت

ویژگی ZoneManagement رابطی برای مدیریت مناطق دلخواه (مناطق فعالیت) در دستگاه‌های دوربین و زنگ در فراهم می‌کند. این مناطق برای فیلتر کردن تشخیص رویداد (مانند حرکت شخص یا وسیله نقلیه) به مناطق خاص در میدان دید دستگاه استفاده می‌شوند.

مناطق فعالیت توسط کاربر در یک برنامه همکار پیکربندی می‌شوند و به آنها امکان می‌دهند مناطق را روی نواحی خاص در میدان دید دوربین ترسیم کنند. این مناطق تعریف شده توسط کاربر سپس به ساختارهای مورد استفاده این ویژگی ترجمه می‌شوند. برای اطلاعات بیشتر در مورد نحوه عملکرد مناطق فعالیت، به بخش «تنظیم و استفاده از مناطق فعالیت» مراجعه کنید.

مناطق فعالیت معمولاً با استفاده از مختصات دکارتی دوبعدی تعریف می‌شوند. این ویژگی، TwoDCartesianVertexStruct را برای رئوس و TwoDCartesianZoneStruct را برای تعریف منطقه (نام، رئوس، رنگ و کاربرد) فراهم می‌کند.

بررسی مناطق فعالیت

برای نمایش مناطق فعالیت، ویژگی zones از ویژگی ZoneManagement را بررسی کنید.

// 1. Obtain the trait flow from the device
private val zoneManagementFlow: Flow =
  device.type(CAMERA_TYPE).flatMapLatest { it.trait(ZoneManagement) }

// 2. Map the flow to the list of zone structures
val activityZones: Flow<List<ZoneManagementTrait.ZoneInformationStruct>> =
  zoneManagementFlow.map { trait ->
    trait.zones ?: emptyList()
  }

اضافه کردن منطقه فعالیت

برای ایجاد یک منطقه جدید، از دستور createTwoDCartesianZone استفاده کنید. این دستور یک TwoDCartesianZoneStruct می‌گیرد که نام، رئوس، رنگ و کاربرد منطقه را تعریف می‌کند.

مثال زیر نحوه ایجاد منطقه‌ای با نام "ایوان جلویی" با چهار رأس، به رنگ ماهی سالمون (#F439A0) و مورد استفاده برای تشخیص حرکت را نشان می‌دهد.

import com.google.home.google.ZoneManagement
import com.google.home.google.ZoneManagementTrait
import com.google.home.matter.serialization.OptionalValue

/**
 * Creates a custom activity zone named "Front Porch" with a salmon color
 * configured for motion detection.
 */
suspend fun createFrontPorchZone(zoneManagement: ZoneManagement) {
  // 1. Define the vertices for the zone (2D Cartesian coordinates)
  // Values are typically scaled to a maximum defined by the device's twoDCartesianMax attribute.
  val vertices =
    listOf(
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 260u, y = 422u),
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 1049u, y = 0u),
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 2048u, y = 0u),
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 2048u, y = 950u),
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 1630u, y = 1349u),
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 880u, y = 2048u),
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 0u, y = 2048u),
      ZoneManagementTrait.TwoDCartesianVertexStruct(x = 638u, y = 1090u)
    )

  // 2. Define the zone structure
  val newZone =
    ZoneManagementTrait.TwoDCartesianZoneStruct(
      name = "Front Porch",
      vertices = vertices,
      // Usage defines what the zone filters (for example, Motion, Person, Vehicle)
      use = listOf(ZoneManagementTrait.ZoneUseEnum.Motion),
      // Color is typically a hex string (for example, Salmon/Pink)
      color = OptionalValue.present("#F439A0")
    )

  try {
    // 3. Execute the command to add the zone to the device
    zoneManagement.createTwoDCartesianZone(newZone)
    println("Successfully created activity zone.")
  } catch (e: Exception) {
    // Handle potential HomeException or Timeout
    println("Failed to create activity zone: ${e.message}")
  }
}

به‌روزرسانی منطقه فعالیت

برای به‌روزرسانی یک منطقه‌ی موجود، از دستور updateTwoDCartesianZone استفاده کنید. این دستور به zoneId و TwoDCartesianZoneStruct به‌روزرسانی‌شده نیاز دارد.

private suspend fun ZoneManagement.updateZone(
  zoneId: UShort,
  zone: ZoneManagementTrait.TwoDCartesianZoneStruct
) {
  // Execute the command to update the zone
  this.updateTwoDCartesianZone(zoneId = zoneId, zone = zone)
}

حذف یک منطقه فعالیت

برای حذف یک منطقه، از دستور removeZone به همراه zoneId مربوطه استفاده کنید.

private suspend fun ZoneManagement.deleteZone(zoneId: UShort) {
  // Execute the command to remove the zone
  this.removeZone(zoneId = zoneId)
}

محرک‌های رویداد صوتی

ویژگی AvStreamAnalysis رابطی برای مدیریت محرک‌های تشخیص رویداد در دستگاه‌های دوربین و زنگ در فراهم می‌کند. در حالی که محرک‌های مبتنی بر بینایی (مانند افراد یا وسایل نقلیه) می‌توانند مختص به منطقه باشند، محرک‌های مرتبط با صدا معمولاً پیکربندی‌هایی در سطح دستگاه هستند.

انواع تریگر زیر برای تشخیص صدا با EventTriggerTypeEnum در دسترس هستند:

حالت مقدار شمارشی توضیحات
صدا Sound تشخیص عمومی صدا
شخص در حال صحبت کردن PersonTalking گفتار را تشخیص می‌دهد.
پارس سگ DogBark صداهای سگ را تشخیص می‌دهد.
شکست شیشه GlassBreak صدای شکستن شیشه را تشخیص می‌دهد.
دزدگیر دود SmokeAlarm آلارم‌های دود را تشخیص می‌دهد، که اغلب با الگوی صوتی T3 (سه بوق کوتاه و به دنبال آن یک مکث) شناخته می‌شوند.
هشدار CO CoAlarm آلارم‌های مونوکسید کربن (CO) را تشخیص می‌دهد، که معمولاً با الگوی صوتی T4 (چهار بوق کوتاه و به دنبال آن مکث) شناخته می‌شوند.

بررسی وضعیت تشخیص صدا

برای نمایش وضعیت فعلی تشخیص صدا به کاربر، باید بررسی کنید که دستگاه از چه چیزی پشتیبانی می‌کند و چه چیزی توسط سخت‌افزار دستگاه فعال شده است. دو ویژگی که باید بررسی شوند عبارتند از:

در توسعه اندروید با استفاده از Kotlin Flows، معمولاً می‌توانید ویژگی AvStreamAnalysis را از HomeDevice مشاهده کنید.

// Example structure to store the
data class EventTriggerAttribute(val type: EventTriggerTypeEnum, val enabled: Boolean)

// 1. Obtain the trait flow from the device
private val avStreamAnalysisFlow: Flow<AvStreamAnalysis> =
  device.traitFromType(AvStreamAnalysis, CAMERA_TYPES.first { device.has(it) })

// 2. Map the flow to a list of sound event attributes
val soundEventTriggersState: Flow<List<EventTriggerAttribute>> =
  avStreamAnalysisFlow.map { trait ->
    // Get raw lists from the trait attributes
    val supported = trait.supportedEventTriggers ?: emptyList()
    val enabled = trait.enabledEventTriggers ?: emptyList()

    // Define sound-specific triggers to filter for
    val soundTypes = setOf(
      EventTriggerTypeEnum.Sound,
      EventTriggerTypeEnum.PersonTalking,
      EventTriggerTypeEnum.DogBark,
      EventTriggerTypeEnum.GlassBreak,
      EventTriggerTypeEnum.SmokeAlarm,
      EventTriggerTypeEnum.CoAlarm,
    )

    // Filter and associate status
    supported
      .filter { soundTypes.contains(it) }
      .map { type ->
        EventTriggerAttribute(
          type = type,
          enabled = enabled.contains(type)
        )
      }
  }

به‌روزرسانی مجموعه تریگرهای فعال‌شده

برای به‌روزرسانی مجموعه‌ی تریگرهای فعال‌شده، از دستور SetOrUpdateEventDetectionTriggers استفاده کنید که فهرستی از ساختارهای EventTriggerEnablement را دریافت می‌کند.

private suspend fun AvStreamAnalysis.updateEventTriggers(
  eventTriggers: List<EventTriggerAttribute>
) {
  val toUpdate = eventTriggers.map {
    EventTriggerEnablement(
      eventTriggerType = it.type,
      enablementStatus = if (it.enabled) {
        EnablementStatusEnum.Enabled
      } else {
        EnablementStatusEnum.Disabled
      },
    )
  }

  // Execute the command on the device
  setOrUpdateEventDetectionTriggers(toUpdate)
}

حالت‌های ضبط

ویژگی RecordingMode رابطی برای مدیریت رفتار ضبط ویدیو و تصویر در دستگاه‌های دوربین و زنگ در فراهم می‌کند. این ویژگی به کاربران امکان می‌دهد بین ضبط مداوم، ضبط مبتنی بر رویداد یا غیرفعال کردن کامل ضبط (فقط نمای زنده) یکی را انتخاب کنند.

RecordingModeEnum استراتژی‌های ضبط موجود را تعریف می‌کند:

حالت مقدار شمارشی توضیحات
معلول Disabled ضبط کاملاً غیرفعال است. عمدتاً توسط دستگاه‌های قدیمی استفاده می‌شود.
CVR (ضبط مداوم ویدیو) Cvr ویدیو به صورت ۲۴ ساعته و ۷ روز هفته ضبط می‌شود. نیاز به اشتراک دارد (برای مثال، Google Google Home Premium .)
EBR (ضبط مبتنی بر رویداد) Ebr ضبط توسط رویدادها (شخص، حرکت) آغاز می‌شود. طول ویدیو به مدت زمان رویداد و اشتراک بستگی دارد.
ETR (ضبط بر اساس رویداد) Etr ضبط پیش‌نمایش کوتاه (مثلاً ۱۰ ثانیه) که توسط رویدادها فعال می‌شود.
نمایش زنده LiveView ضبط غیرفعال است، اما کاربران همچنان می‌توانند به پخش زنده دسترسی داشته باشند.
تصاویر ثابت Images هنگام وقوع رویدادها، به جای فیلم، عکس‌های فوری ضبط می‌شوند.

بررسی حالت‌های ضبط

برای نمایش پیکربندی ضبط فعلی، ویژگی‌های ویژگی RecordingMode را بررسی کنید:

// 1. Obtain the trait flow from the device
private val recordingModeTraitFlow: Flow =
    device.traitFromType(RecordingMode, CAMERA_TYPES.first { device.has(it) })

// 2. Map the flow to recording mode options
data class RecordingModeOptions(
    val recordingMode: RecordingModeTrait.RecordingModeEnum,
    val index: Int,
    val available: Boolean,
    val readableString: String,
)

private val recordingModeOptions: Flow<List> =
    recordingModeTraitFlow.map { trait ->
        val supported = trait.supportedRecordingModes?.map { it.recordingMode } ?: emptyList()
        val available = trait.availableRecordingModes?.map { it.toInt() } ?: emptyList()

        supported.withIndex().map { (index, mode) ->
            RecordingModeOptions(
                recordingMode = mode,
                index = index,
                available = available.contains(index),
                readableString = mode.toReadableString(),
            )
        }
    }

تغییر حالت ضبط

قبل از به‌روزرسانی، مطمئن شوید که اندیس انتخاب‌شده از ویژگی supportedRecordingModes در ویژگی availableRecordingModes موجود باشد.

برای به‌روزرسانی حالت انتخاب‌شده، از تابع setSelectedRecordingMode استفاده کنید و اندیس حالت انتخاب‌شده را به آن ارسال کنید:

private suspend fun RecordingMode.updateRecordingMode(index: Int) {
    // Execute the command to update the selected mode
    this.setSelectedRecordingMode(index.toUByte())
}

تنظیمات دیگر

تنظیمات مختلف دیگری را می‌توان از طریق APIهای Home کنترل کرد.

تغییر جهت تصویر

جهت تصویر دوربین (فیلم) قابل چرخش است. فیلم فقط ۱۸۰ درجه قابل چرخش است.

برای تغییر جهت تصویر دوربین، ویژگی imageRotation از ویژگی CameraAvStreamManagement را با استفاده از تابع داخلی setImageRotation کاتلین به‌روزرسانی کنید:

// Change the camera's image orientation
val isRotated = false

cameraAvStreamManagement.update { setImageRotation(if (isRotated) 180.toUShort() else 0.toUShort()) }

فعال یا غیرفعال کردن دید در شب

برای فعال یا غیرفعال کردن دید در شب برای دوربین، از TriStateAutoEnum برای به‌روزرسانی ویژگی nightVision از ویژگی CameraAvStreamManagement با استفاده از تابع داخلی setNightVision کاتلین استفاده کنید:

// Turn night vision on
cameraAvStreamManagement.update {
  setNightVision(CameraAvStreamManagementTrait.TriStateAutoEnum.On)
}

// Turn night vision off
CameraAvStreamManagement.update {
  setNightVision(CameraAvStreamManagementTrait.TriStateAutoEnum.Off)
}

تغییر روشنایی LED وضعیت

برای تغییر روشنایی LED وضعیت، ThreeLevelAutoEnum برای به‌روزرسانی ویژگی statusLightBrightness از ویژگی CameraAvStreamManagement با استفاده از تابع داخلی setStatusLightBrightness کاتلین استفاده کنید:

// Set the LED brightness to high
cameraAvStreamManagement.update {
  setStatusLightBrightness(CameraAvStreamManagementTrait.ThreeLevelAutoEnum.High)
}

// Set the LED brightness to low
cameraAvStreamManagement.update {
  setStatusLightBrightness(CameraAvStreamManagementTrait.ThreeLevelAutoEnum.Low)
}

تغییر زاویه دید دوربین

نمای دوربین همان ویژگی بزرگنمایی و برش است که در مقاله پشتیبانی از ویدیوی دوربین Nest با بزرگنمایی و بهبود، توضیح داده شده است.

نمای دید (viewport) در یک ViewportStruct تعریف شده است که شامل چهار مقدار است که به عنوان مختصات نمای دید استفاده می‌شوند. این مختصات به صورت زیر تعریف می‌شوند:

(x1,y1) -- (x2,y1)
   |          |
(x1,y2) -- (x2,y2)

تعیین مقادیر ViewportStruct به رابط کاربری برنامه و پیاده‌سازی دوربین بستگی دارد. در سطح بسیار ابتدایی، برای تنظیم نمای ویدیوی دوربین، ویژگی viewport از ویژگی CameraAvStreamManagement را با ViewportStruct به‌روزرسانی کنید، و از تابع داخلی setViewport کاتلین استفاده کنید:

cameraAvStreamManagement
  .update { setViewport(
    CameraAvStreamManagementTrait.ViewportStruct(
      x1 = horizontalRange.rangeStart.roundToInt().toUShort(),
      x2 = horizontalRange.rangeEnd.roundToInt().toUShort(),
      y1 = verticalRange.rangeStart.roundToInt().toUShort(),
      y2 = verticalRange.rangeEnd.roundToInt().toUShort(),
    )
) }

حساسیت بیدار شدن دستگاه را تنظیم کنید

حساسیت بیدار شدن دستگاه برای صرفه‌جویی در مصرف باتری استفاده می‌شود، به این صورت که محدوده‌ای که دستگاه می‌تواند فعالیت را حس کند کاهش می‌یابد و زمان بیدار شدن پس از تشخیص آن فعالیت افزایش می‌یابد.

در APIهای Home، این مورد را می‌توان با استفاده از ویژگی motionSensitivity از triggerOptions در transportOptions دستگاه تنظیم کرد. این گزینه‌ها در ویژگی PushAvStreamTransport برای هر دستگاه تعریف شده‌اند.

حساسیت بیدارباش فقط می‌تواند روی مقادیر زیر تنظیم شود:

  • ۱ = کم
  • ۵ = متوسط
  • ۱۰ = زیاد

فرآیند به‌روزرسانی به این صورت است که پیکربندی انتقال برای جریان‌های ضبط فعال با استفاده از دستور findTransport پیدا می‌شود، سپس پیکربندی با مقدار حساسیت جدید با استفاده از دستور modifyPushTransport اصلاح می‌شود:

// Create a struct with the new wake-up sensitivity
val toUpdate =  TransportOptionsStruct(
  triggerOptions =
    TransportTriggerOptionsStruct(
      motionSensitivity =
        OptionalValue.present(wakeUpSensitivity.toUByte())
    )
  )

// Get the configurations for active connections
val connections  = pushAvStreamTransport.findTransport().transportConfigurations
  // Update all recording streams with the new transport options.
  for (connection in connections) {
    if (connection.transportOptions.getOrNull()?.streamUsage == StreamUsageEnum.Recording) {
      trait.modifyPushTransport(
        connectionId = connection.connectionId,
        transportOptions = toUpdate,
      )
    }
  }

حداکثر طول رویداد را تنظیم کنید

حداکثر طول رویداد، مدت زمانی است که دوربین کلیپی از یک رویداد ضبط می‌کند. از طریق APIهای Home، این مدت زمان را می‌توان برای هر دستگاه، با همان طول زمانی که از طریق GHA و در فواصل ثانیه‌ای تنظیم می‌شود، تنظیم کرد:

  • ۱۰ ثانیه
  • ۱۵ ثانیه
  • ۳۰ ثانیه
  • ۶۰ ثانیه (۱ دقیقه)
  • ۱۲۰ ثانیه (۲ دقیقه)
  • ۱۸۰ ثانیه (۳ دقیقه)

در APIهای Home، این مورد را می‌توان با استفاده از ویژگی motionTimeControl از triggerOptions در transportOptions دستگاه تنظیم کرد. این گزینه‌ها در ویژگی PushAvStreamTransport برای هر دستگاه تعریف شده‌اند.

فرآیند به‌روزرسانی به این صورت است که پیکربندی انتقال برای جریان‌های ضبط فعال با استفاده از دستور findTransport پیدا می‌شود، سپس پیکربندی با مقدار طول رویداد جدید با استفاده از دستور modifyPushTransport اصلاح می‌شود:

// Create a struct with the new max event length
// where maxDuration is the length in seconds
val toUpdate =  TransportOptionsStruct(
  triggerOptions =
    TransportTriggerOptionsStruct(
      motionTimeControl =
        OptionalValue.present(
          TransportMotionTriggerTimeControlStruct(maxDuration = it.toUInt())
        )
    )
  )

// Get the configurations for active connections
val connections  = pushAvStreamTransport.findTransport().transportConfigurations
  // Update all recording streams with the new transport options.
  for (connection in connections) {
    if (connection.transportOptions.getOrNull()?.streamUsage == StreamUsageEnum.Recording) {
      trait.modifyPushTransport(
        connectionId = connection.connectionId,
        transportOptions = toUpdate,
      )
    }
  }

فعال یا غیرفعال کردن تجزیه و تحلیل

هر دستگاه می‌تواند به صورت جداگانه ارسال داده‌های تحلیلی دقیق به فضای ابری گوگل هوم را انتخاب کند (به بخش نظارت ابری برای APIهای خانگی مراجعه کنید).

برای فعال کردن تجزیه و تحلیل برای یک دستگاه، ویژگی analyticsEnabled از ExtendedGeneralDiagnosticsTrait را روی true تنظیم کنید. وقتی analyticsEnabled را تنظیم می‌کنید، ویژگی دیگری به نام logUploadEnabled به طور خودکار روی true تنظیم می‌شود که به فایل‌های گزارش تجزیه و تحلیل اجازه می‌دهد تا در فضای ابری Google Home آپلود شوند.

// Enable analytics
extendedGeneralDiagnostics.update {
  setAnalyticsEnabled(true)
}

// Disable analytics
extendedGeneralDiagnostics.update {
  setAnalyticsEnabled(false)
}