| name | upsert-case |
| description | Add a new API example case or modify an existing one in the APIExample Android demo — creates or updates Fragment class, XML layout, string resources, and nav_graph registration. Use when: adding a new Agora RTC API demo screen, modifying an existing case's implementation or registration, implementing a new feature example in Java + XML layouts, registering a new case via @Example annotation, subclassing BaseFragment for a new demo screen, or updating an existing case's strings, layout, or nav entry. Keywords: add case, modify case, update case, new fragment, nav_graph, @Example, BaseFragment, APIExample, new screen, demo case, RTC API example.
|
Upsert Case — APIExample
Adding a New Case
Touch exactly 4 files (all paths relative to app/src/main/):
| File | What to add |
|---|
java/.../examples/{basic|advanced|audio}/YourCaseName.java | Fragment class |
res/layout/fragment_your_case_name.xml | XML layout |
res/values/strings.xml | 2 strings |
res/navigation/nav_graph.xml | 1 action + 1 destination |
Registration is automatic via reflection — no other files needed.
Step 1: Clarify before coding
Before writing a single line, ask:
- What API am I demonstrating? — determines which existing case is the closest reference to copy patterns from
- Video or audio-only? — determines permissions (
CAMERA + RECORD_AUDIO vs RECORD_AUDIO only), layout complexity, and whether VideoReportLayout is needed
- BASIC or ADVANCED group? — BASIC for fundamental channel join/leave patterns; ADVANCED for feature-specific APIs
- What's the sort index? — index must be unique within the group. BASIC uses 0–9, ADVANCED starts from 10. Run
query-cases skill first; a collision causes silent ordering bugs at runtime
Step 2: Create the Fragment
MANDATORY — READ ENTIRE FILE before writing any code:
references/fragment-template.java
Do NOT skip — the setParameters, handler.post, and getPrivateCloudConfig() null-check patterns are only fully shown there and are required in every case.
Do NOT load any other reference files for this task.
Non-obvious points the template highlights:
setParameters(...) for app scenario reporting — required in every case, do not remove
handler.post(RtcEngine::destroy) — NOT RtcEngine.destroy() directly; direct call blocks UI thread (ANR)
getPrivateCloudConfig() null-check before setLocalAccessPoint() — returns null on non-private-cloud builds (NPE)
- All
IRtcEngineEventHandler callbacks run on a background thread — always runOnUIThread() for UI
onActivityCreated → create engine; onDestroy → leaveChannel() then handler.post(RtcEngine::destroy)
For video cases, add VideoReportLayout fields and wire setupRemoteVideo in onUserJoined/onUserOffline.
Step 3: Create the XML layout
Minimum structure — channel input + join button at bottom:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitsSystemWindows="true">
<LinearLayout
android:id="@+id/ll_join"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:gravity="center_vertical"
android:orientation="horizontal">
<androidx.appcompat.widget.AppCompatEditText
android:id="@+id/et_channel"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:digits="@string/chanel_support_char"
android:hint="@string/channel_id" />
<androidx.appcompat.widget.AppCompatButton
android:id="@+id/btn_join"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/join" />
</LinearLayout>
</RelativeLayout>
For video cases, use VideoReportLayout for each video slot. Pick one of the four standard layouts below — they cover the vast majority of cases.
General rules (apply to all layouts):
- Video containers must sit above the bottom control bar. In
RelativeLayout use android:layout_above="@id/ll_join"; in ConstraintLayout use app:layout_constraintBottom_toTopOf="@id/ll_join".
- Each
VideoReportLayout needs a unique android:id (fl_local, fl_remote, fl_remote2, …).
Layout A — Single broadcaster (local fullscreen)
Use when: broadcaster-only demo, no remote video needed.
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_local"
android:layout_width="match_parent"
android:layout_height="0dp"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintBottom_toTopOf="@id/ll_join" />
Layout B — 1v1 (local left, remote right, side by side)
Use when: two-party call, equal-weight split.
<LinearLayout
android:id="@+id/video_container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_above="@id/ll_join"
android:orientation="horizontal">
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_local"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_weight="0.5" />
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_remote"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_weight="0.5" />
</LinearLayout>
Layout C — Audience co-hosting (remote fullscreen background + local PiP top-right)
Use when: live streaming where audience co-hosts; remote/host fills screen, local is a small overlay.
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_remote"
android:layout_width="match_parent"
android:layout_height="0dp"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintBottom_toTopOf="@id/ll_join" />
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_local"
android:layout_width="120dp"
android:layout_height="160dp"
android:layout_margin="8dp"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
Layout D — 2×2 grid (up to 4 participants)
Use when: multi-party call with up to 4 streams.
<LinearLayout
android:id="@+id/video_container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_above="@id/ll_join"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="0.5"
android:orientation="horizontal">
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_local"
android:layout_width="0dp" android:layout_height="match_parent"
android:layout_weight="0.5" />
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_remote"
android:layout_width="0dp" android:layout_height="match_parent"
android:layout_weight="0.5" />
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="0.5"
android:orientation="horizontal">
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_remote2"
android:layout_width="0dp" android:layout_height="match_parent"
android:layout_weight="0.5" />
<io.agora.api.example.common.widget.VideoReportLayout
android:id="@+id/fl_remote3"
android:layout_width="0dp" android:layout_height="match_parent"
android:layout_weight="0.5" />
</LinearLayout>
</LinearLayout>
Step 4: Add nav entries
File: res/navigation/nav_graph.xml
Action — inside <fragment android:id="@+id/Ready"> (NOT mainFragment — mainFragment only has one action, to Ready):
<action
android:id="@+id/action_mainFragment_to_yourCaseName"
app:destination="@id/yourCaseName" />
Destination — at root <navigation> level:
<fragment
android:id="@+id/yourCaseName"
android:name="io.agora.api.example.examples.advanced.YourCaseName"
android:label="@string/item_your_case_name"
tools:layout="@layout/fragment_your_case_name" />
action android:id must exactly match actionId in @Example.
Step 5: Update ARCHITECTURE.md
Add one line to the case list in ARCHITECTURE.md under the correct directory section (basic/, advanced/, or audio/):
├── YourCaseName.java # [index] "Display Name" — key API description
Keep the format consistent with existing entries. This file is the fast-lookup index used by query-cases — keeping it current avoids full directory scans.
Modifying an Existing Case
When modifying an existing case rather than creating a new one, identify which files need changes based on what you are updating:
| What changed | Files to touch |
|---|
| Implementation logic (API calls, event handling) | java/.../examples/{basic|advanced|audio}/CaseName.java |
| UI layout (views, controls, video containers) | res/layout/fragment_case_name.xml |
| Display name or tips text | res/values/strings.xml |
| Sort index or group (BASIC ↔ ADVANCED) | @Example annotation in the Fragment class |
| Navigation label | res/navigation/nav_graph.xml (fragment label attribute) |
| Class rename or package move | Fragment class, nav_graph.xml (android:name + destination id), @Example annotation (actionId), layout file name, ARCHITECTURE.md |
After making changes:
- Verify
@Example annotation consistency — ensure index, group, name, actionId, and tipsId still match the actual string resources, nav action ID, and intended group/position. A mismatch causes the case to silently disappear from the list or navigate to the wrong screen.
- Update
res/values/strings.xml if the display name or tips text changed.
- Update
res/navigation/nav_graph.xml if the class name, package, or label changed.
- Update
ARCHITECTURE.md — update the Directory Layout entry and the Case Index table row to reflect any changes to the case name, path, Key APIs, or description.
Verify
./gradlew assembleDebug
When to Use a Spec Instead
If the case meets any of the following criteria, create a Spec rather than using this skill directly:
- Involves coordinated calls across two or more Agora API modules
- Requires a custom UI layout (not one of the standard Layout A/B/C/D templates above)
- Involves multi-channel or multi-engine instance management
- Requires a foreground Service or background thread coordination
- Involves developing new shared components (widgets/utils, etc.)
- Requires optional module integration (simpleFilter/streamEncrypt)
If none apply → use this skill directly; no Spec needed.
Spec Requirements Document Must Include
- List of APIs the case demonstrates
- User interaction flow description
- Expected RtcEngine lifecycle behavior
- Required permissions list
Spec Design Document Must Include
- Target project identifier:
APIExample
- Class/file structure design
- API call sequence (Mermaid sequence diagram recommended)
- State management approach
- UI layout plan
- Integration points with existing shared components
- Case registration info: class name, display name, group (BASIC/ADVANCED), sort index — finalize during design to avoid conflicts
- Generate
@Example annotation parameters, nav_graph.xml action + destination, strings.xml key names (item_ prefix)
- Read
ARCHITECTURE.md or use the query-cases skill to check existing indices
- Risk identification and mitigation (API compatibility, performance, permissions, thread safety)
Spec Task List Integration
- Mark which sub-tasks can be executed with this
upsert-case skill, and provide skill input parameters
- Mark which sub-tasks require manual coding, and provide target file paths and change summaries
- Tasks for creating new shared components must come before case implementation tasks
NEVER
- NEVER put the nav action inside
<fragment id="mainFragment"> — it belongs in <fragment id="Ready">. mainFragment only routes to Ready; all case actions live in Ready. Wrong placement causes silent navigation failure at runtime.
- NEVER call
RtcEngine.destroy() directly on the main thread — always handler.post(RtcEngine::destroy). Direct call blocks the UI thread and causes ANR.
- NEVER call
setLocalAccessPoint() without null-checking getPrivateCloudConfig() first — it returns null on standard builds, causing NPE.
- NEVER update UI directly inside
IRtcEngineEventHandler callbacks — they run on a background thread. Always wrap with runOnUIThread().
- NEVER omit
setParameters(...) — it's required for Agora backend usage reporting in every case; omitting it causes silent reporting failure even though the app appears to work normally.