mob_camera

Native camera capture, live preview, and frame streaming for apps built with MobMob.Camera, extracted from mob core as a plugin.

iOS: UIImagePickerController + a shared AVCaptureSession (vImage frame conversion). Android: TakePicture/CaptureVideo activity contracts + CameraX ImageAnalysis.

Installation

# mix.exs
{:mob_camera, "~> 0.1"}
# mob.exs
config :mob, :plugins, [:mob_camera]

The plugin manifest merges NSCameraUsageDescription (iOS) and CAMERA / RECORD_AUDIO (Android) into the host app at build time, and registers the :camera permission capability — request it via Mob.Permissions.request(socket, :camera) before capturing. (:microphone, needed for video, stays in core.)

Usage

socket = MobCamera.capture_photo(socket, quality: :high)
socket = MobCamera.capture_video(socket, max_duration: 60)
def handle_info({:camera, :photo, %{path: path, width: w, height: h}}, socket), do: ...
def handle_info({:camera, :video, %{path: path, duration: seconds}}, socket), do: ...
def handle_info({:camera, :cancelled}, socket), do: ...

path is a local temp file — copy it elsewhere before the next capture.

For real-time work (object detection, AR, custom filters), stream frames:

socket = MobCamera.start_frame_stream(socket, width: 640, height: 640, format: :rgb_f32)
def handle_info({:camera, :frame, %{bytes: bin, width: w, height: h,
format: :rgb_f32, timestamp_ms: _t,
dropped: _n}}, socket) do
# :rgb_f32 is Nx-ready: Nx.from_binary(bin, :f32) |> Nx.reshape({1, h, w, 3})
end

Resize + format conversion happen natively, and late frames are dropped natively, so the BEAM mailbox stays bounded. Other options: format: :bgra_u8, facing: :front, throttle_ms:. Stop with MobCamera.stop_frame_stream/1.

Live preview pairs a session from this plugin with a view component from core:

socket = MobCamera.start_preview(socket, facing: :back)
# in render/1:
{Mob.UI.camera_preview(facing: :back)}

Host app requirements

Photo/video capture saves through a FileProvider: AndroidManifest.xml must declare an androidx.core.content.FileProvider<provider> with res/xml/file_provider_paths.xml. mob_new-generated apps include it; hand-rolled hosts must add it or capture returns :cancelled.

Limits

Development

Clone, then run once:

mix setup

That fetches deps and activates the repo's git hooks (.githooks/pre-push): mix format --check, mix credo --strict (incl. ExSlop), and mix compile --warnings-as-errors run on every push, plus the full test suite when mix.exs changes — the same gate CI enforces before publishing.

License

MIT