Розробка мобільного додатку для управління коворкингом
Коворкинг—це кілька типів простору з різними правилами доступу: відкриті зони по абонементу, переговорні по брой, приватні офіси по довгостроковому договору. Мобільний додаток повинен охоплювати весь цей спектр, плюс—стати ключем доступу (замість фізичних карт або кодів на двері).
Бронювання переговорних: часові слоти
Переговорні бронюються на конкретний час. Головне завдання UX—показати вільні слоти наглядно та обробити конкурентні бронювання:
// Android: тайм-слоти з Jetpack Compose
@Composable
fun TimeSlotPicker(
slots: List<TimeSlot>,
onSlotSelected: (TimeSlot) -> Unit
) {
LazyRow(horizontalArrangement = Arrangement.spacedBy(8.dp)) {
items(slots) { slot ->
TimeSlotChip(
startTime = slot.startTime,
endTime = slot.endTime,
isAvailable = slot.status == SlotStatus.AVAILABLE,
isSelected = slot.isSelected,
onClick = {
if (slot.status == SlotStatus.AVAILABLE) onSlotSelected(slot)
}
)
}
}
}
Слоти завантажуються через GET /rooms/{roomId}/slots?date=2024-09-21. Інтервал опитування при відкритому екрані бронювання—30 секунд, щоб відображати актуальність. Або WebSocket, якщо важлива миттєва реакція.
Цифровий доступ через мобільний ключ
Основна «фіча» сучасних коворкингів—смартфон замість RFID-карти. Реалізується трьома способами:
QR-код на двері—додаток показує динамічний QR (оновлюється щомісяця 30 секунд), статичний ридер на двері його сканує. Найпростіший варіант.
NFC-емуляція (Host Card Emulation на Android)—смартфон емулює RFID-карту. Користувач прикладає телефон до зчитувача.
// Android: HCE для доступу
class AccessHCEService : HostApduService() {
override fun processCommandApdu(commandApdu: ByteArray, extras: Bundle?): ByteArray {
val selectAid = byteArrayOf(0x00, 0xA4.toByte(), 0x04, 0x00)
return if (commandApdu.startsWith(selectAid)) {
val accessToken = accessRepository.getCurrentToken()
accessToken.toByteArray() + byteArrayOf(0x90.toByte(), 0x00)
} else {
byteArrayOf(0x6F, 0x00) // невідома команда
}
}
override fun onDeactivated(reason: Int) { }
}
HCE працює без NFC-ридера Apple Pay—лише через власний Android NFC stack. На iOS аналог—Core NFC з NFCNDEFReaderSession, але для запису/емуляції карт iOS закритий. Тому на iOS—QR або Bluetooth BLE.
Bluetooth BLE—замки з BLE-чипом (популярні у Salto, Abloy). Додаток відправляє зашифровану команду на замок:
import CoreBluetooth
class BLELockManager: NSObject, CBCentralManagerDelegate, CBPeripheralDelegate {
func openLock(peripheral: CBPeripheral, accessToken: String) {
guard let characteristic = lockCharacteristic else { return }
let payload = buildAccessPayload(token: accessToken)
peripheral.writeValue(payload, for: characteristic, type: .withResponse)
}
func peripheral(_ peripheral: CBPeripheral,
didWriteValueFor characteristic: CBCharacteristic,
error: Error?) {
if error == nil {
hapticFeedback.notificationOccurred(.success)
// Двері відкрилися
}
}
}
Абонементи та списання
Абонемент на коворкинг—пакет годин або днів. Списання при кожному візиту:
- Почасовий абонемент: time in/time out, списуємо
ceil(minutes / 60)годин - Денний абонемент: один день за будь-який візит у рамках 24 годин
- Безліміт: лише перевірка терміну дії
Баланс абонемента оновлюється в реальному часі—через push після закриття сесії. Якщо баланс закінчився на половині дня—пропонуємо докупити прямо з повідомлення.
Система резидентів та гостей
Довгостроковые резиденти мають постійний доступ та розширені права (зберігання речей в ячейках, виділений принтер). Гості—разовий доступ по QR, який резидент генерує для колеги з додатку.
Гостевой QR—обмежено: лише конкретна зона, конкретна дата, максимум один вхід:
def generate_guest_qr(
host_user_id: str,
zones: list[str],
valid_date: date
) -> str:
payload = {
"type": "GUEST",
"hostId": host_user_id,
"zones": zones,
"validDate": valid_date.isoformat(),
"singleUse": True,
"token": secrets.token_urlsafe(16)
}
return sign_and_encode(payload, private_key)
Ориентири по терміне
Базова версія (каталог просторів, бронювання переговорних, QR-доступ, абонементи): 5–7 тижнів. BLE-інтеграція з конкретними замками—ще 1–2 тижні залежно від виробника. Вартість розраховується індивідуально.







