Install
openclaw skills install questions-formPresent multiple clarifying questions as an interactive Telegram form using inline buttons. Use when the agent needs to ask the user 2 or more clarifying questions before proceeding with a task, and wants to present them all at once in a structured form layout with selectable options and an "Other" free-text escape hatch. Triggers when: gathering multi-faceted requirements, onboarding flows, preference collection, or any scenario requiring structured multi-question input from the user via Telegram.
openclaw skills install questions-formPresent multiple clarifying questions as a Telegram inline-button form. All questions appear at once; the user answers in any order, then submits.
Do not use this pattern for a single yes/no question — just send one message with buttons for that.
Define each question internally as an object:
{ id: "type", text: "What type of project?", options: ["Web App", "Mobile", "API"] }
{ id: "timeline", text: "What is your timeline?", options: ["This week", "This month", "No rush"] }
{ id: "budget", text: "Budget range?", options: ["< $1k", "$1k-5k", "$5k-10k", "> $10k"] }
Initialize internal tracking state:
form_state = { type: null, timeline: null, budget: null }
awaiting_freetext = null
form_submitted = false
Send a plain message (no buttons) as the form introduction:
{
"action": "send",
"channel": "telegram",
"message": "**I have a few questions before we proceed.**\nPlease answer each one by tapping a button, then tap Submit when done."
}
For each question, send a separate message with inline buttons. Put selectable options in rows of 2-3 buttons. Always put "Other" on its own last row.
The callback_data must follow this convention: form:<question_id>:<value>
Example:
{
"action": "send",
"channel": "telegram",
"message": "**1. What type of project is this?**",
"buttons": [
[
{ "text": "Web App", "callback_data": "form:type:web" },
{ "text": "Mobile", "callback_data": "form:type:mobile" },
{ "text": "API", "callback_data": "form:type:api" }
],
[
{ "text": "Other (type your answer)", "callback_data": "form:type:other" }
]
]
}
{
"action": "send",
"channel": "telegram",
"message": "**2. What is your timeline?**",
"buttons": [
[
{ "text": "This week", "callback_data": "form:timeline:this_week" },
{ "text": "This month", "callback_data": "form:timeline:this_month" },
{ "text": "No rush", "callback_data": "form:timeline:no_rush" }
],
[
{ "text": "Other (type your answer)", "callback_data": "form:timeline:other" }
]
]
}
{
"action": "send",
"channel": "telegram",
"message": "**3. Budget range?**",
"buttons": [
[
{ "text": "< $1k", "callback_data": "form:budget:lt_1k" },
{ "text": "$1k-5k", "callback_data": "form:budget:1k_5k" }
],
[
{ "text": "$5k-10k", "callback_data": "form:budget:5k_10k" },
{ "text": "> $10k", "callback_data": "form:budget:gt_10k" }
],
[
{ "text": "Other (type your answer)", "callback_data": "form:budget:other" }
]
]
}
After all questions, send the submit/cancel message:
{
"action": "send",
"channel": "telegram",
"message": "**When you've answered all questions above, tap Submit.**",
"buttons": [
[{ "text": "\u2713 Submit All Answers", "callback_data": "form:submit" }],
[{ "text": "\u2717 Cancel", "callback_data": "form:cancel" }]
]
}
When you receive a callback starting with form::
Regular option (form:<qid>:<value> where value is not other):
form_state[qid] = value"Got it -- <question label>: **<chosen label>**""Other" option (form:<qid>:other):
"Type your answer for: <question text>"awaiting_freetext = qidform_state[qid] = <user's text>"Got it -- <question label>: **<user's text>**"awaiting_freetextSubmit (form:submit):
form_state for any null values"You still need to answer: <list of unanswered questions>"form_submitted = true and proceed with the collected answersCancel (form:cancel):
form_state"Form cancelled. Let me know how you'd like to proceed."Once submitted, reference the collected answers as structured data and proceed:
Collected: { type: "mobile", timeline: "End of March", budget: "1k_5k" }
Now continue with the original task using these clarified requirements.
Users can click a different button for any question at any time before submitting. Simply overwrite the previous value and acknowledge the change:
"Updated -- <question>: **<new value>**"
form: prefixform:<question_id>:<value>form:"Other (type your answer)""\u2713 Submit All Answers""\u2717 Cancel"See references/form-patterns.md for: