machinaos 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env.template +71 -0
- package/LICENSE +21 -0
- package/README.md +87 -0
- package/bin/cli.js +159 -0
- package/client/.dockerignore +45 -0
- package/client/Dockerfile +68 -0
- package/client/eslint.config.js +29 -0
- package/client/index.html +13 -0
- package/client/nginx.conf +66 -0
- package/client/package.json +48 -0
- package/client/src/App.tsx +27 -0
- package/client/src/Dashboard.tsx +1173 -0
- package/client/src/ParameterPanel.tsx +301 -0
- package/client/src/components/AIAgentNode.tsx +321 -0
- package/client/src/components/APIKeyValidator.tsx +118 -0
- package/client/src/components/ClaudeChatModelNode.tsx +18 -0
- package/client/src/components/ConditionalEdge.tsx +189 -0
- package/client/src/components/CredentialsModal.tsx +306 -0
- package/client/src/components/EdgeConditionEditor.tsx +443 -0
- package/client/src/components/GeminiChatModelNode.tsx +18 -0
- package/client/src/components/GenericNode.tsx +357 -0
- package/client/src/components/LocationParameterPanel.tsx +154 -0
- package/client/src/components/ModelNode.tsx +286 -0
- package/client/src/components/OpenAIChatModelNode.tsx +18 -0
- package/client/src/components/OutputPanel.tsx +471 -0
- package/client/src/components/ParameterRenderer.tsx +1874 -0
- package/client/src/components/SkillEditorModal.tsx +417 -0
- package/client/src/components/SquareNode.tsx +797 -0
- package/client/src/components/StartNode.tsx +250 -0
- package/client/src/components/ToolkitNode.tsx +365 -0
- package/client/src/components/TriggerNode.tsx +463 -0
- package/client/src/components/auth/LoginPage.tsx +247 -0
- package/client/src/components/auth/ProtectedRoute.tsx +59 -0
- package/client/src/components/base/BaseChatModelNode.tsx +271 -0
- package/client/src/components/icons/AIProviderIcons.tsx +50 -0
- package/client/src/components/maps/GoogleMapsPicker.tsx +137 -0
- package/client/src/components/maps/MapsPreviewPanel.tsx +110 -0
- package/client/src/components/maps/index.ts +26 -0
- package/client/src/components/parameterPanel/InputSection.tsx +1094 -0
- package/client/src/components/parameterPanel/LocationPanelLayout.tsx +65 -0
- package/client/src/components/parameterPanel/MapsSection.tsx +92 -0
- package/client/src/components/parameterPanel/MiddleSection.tsx +571 -0
- package/client/src/components/parameterPanel/OutputSection.tsx +81 -0
- package/client/src/components/parameterPanel/ParameterPanelLayout.tsx +82 -0
- package/client/src/components/parameterPanel/ToolSchemaEditor.tsx +436 -0
- package/client/src/components/parameterPanel/index.ts +42 -0
- package/client/src/components/shared/DataPanel.tsx +142 -0
- package/client/src/components/shared/JSONTreeRenderer.tsx +106 -0
- package/client/src/components/ui/AIResultModal.tsx +204 -0
- package/client/src/components/ui/AndroidSettingsPanel.tsx +401 -0
- package/client/src/components/ui/CodeEditor.tsx +81 -0
- package/client/src/components/ui/CollapsibleSection.tsx +88 -0
- package/client/src/components/ui/ComponentItem.tsx +154 -0
- package/client/src/components/ui/ComponentPalette.tsx +321 -0
- package/client/src/components/ui/ConsolePanel.tsx +1074 -0
- package/client/src/components/ui/ErrorBoundary.tsx +196 -0
- package/client/src/components/ui/InputNodesPanel.tsx +204 -0
- package/client/src/components/ui/MapSelector.tsx +314 -0
- package/client/src/components/ui/Modal.tsx +149 -0
- package/client/src/components/ui/NodeContextMenu.tsx +192 -0
- package/client/src/components/ui/NodeOutputPanel.tsx +1150 -0
- package/client/src/components/ui/OutputDisplayPanel.tsx +381 -0
- package/client/src/components/ui/SettingsPanel.tsx +243 -0
- package/client/src/components/ui/TopToolbar.tsx +736 -0
- package/client/src/components/ui/WhatsAppSettingsPanel.tsx +345 -0
- package/client/src/components/ui/WorkflowSidebar.tsx +294 -0
- package/client/src/config/antdTheme.ts +186 -0
- package/client/src/config/api.ts +54 -0
- package/client/src/contexts/AuthContext.tsx +221 -0
- package/client/src/contexts/ThemeContext.tsx +42 -0
- package/client/src/contexts/WebSocketContext.tsx +1971 -0
- package/client/src/factories/baseChatModelFactory.ts +256 -0
- package/client/src/hooks/useAndroidOperations.ts +164 -0
- package/client/src/hooks/useApiKeyValidation.ts +107 -0
- package/client/src/hooks/useApiKeys.ts +238 -0
- package/client/src/hooks/useAppTheme.ts +17 -0
- package/client/src/hooks/useComponentPalette.ts +51 -0
- package/client/src/hooks/useCopyPaste.ts +155 -0
- package/client/src/hooks/useDragAndDrop.ts +124 -0
- package/client/src/hooks/useDragVariable.ts +88 -0
- package/client/src/hooks/useExecution.ts +313 -0
- package/client/src/hooks/useParameterPanel.ts +176 -0
- package/client/src/hooks/useReactFlowNodes.ts +189 -0
- package/client/src/hooks/useToolSchema.ts +209 -0
- package/client/src/hooks/useWhatsApp.ts +196 -0
- package/client/src/hooks/useWorkflowManagement.ts +46 -0
- package/client/src/index.css +315 -0
- package/client/src/main.tsx +19 -0
- package/client/src/nodeDefinitions/aiAgentNodes.ts +336 -0
- package/client/src/nodeDefinitions/aiModelNodes.ts +340 -0
- package/client/src/nodeDefinitions/androidDeviceNodes.ts +140 -0
- package/client/src/nodeDefinitions/androidServiceNodes.ts +383 -0
- package/client/src/nodeDefinitions/chatNodes.ts +135 -0
- package/client/src/nodeDefinitions/codeNodes.ts +54 -0
- package/client/src/nodeDefinitions/documentNodes.ts +379 -0
- package/client/src/nodeDefinitions/index.ts +15 -0
- package/client/src/nodeDefinitions/locationNodes.ts +463 -0
- package/client/src/nodeDefinitions/schedulerNodes.ts +220 -0
- package/client/src/nodeDefinitions/skillNodes.ts +211 -0
- package/client/src/nodeDefinitions/toolNodes.ts +198 -0
- package/client/src/nodeDefinitions/utilityNodes.ts +284 -0
- package/client/src/nodeDefinitions/whatsappNodes.ts +865 -0
- package/client/src/nodeDefinitions/workflowNodes.ts +41 -0
- package/client/src/nodeDefinitions.ts +104 -0
- package/client/src/schemas/workflowSchema.ts +264 -0
- package/client/src/services/dynamicParameterService.ts +96 -0
- package/client/src/services/execution/aiAgentExecutionService.ts +35 -0
- package/client/src/services/executionService.ts +232 -0
- package/client/src/services/workflowApi.ts +91 -0
- package/client/src/store/useAppStore.ts +582 -0
- package/client/src/styles/theme.ts +508 -0
- package/client/src/styles/zIndex.ts +17 -0
- package/client/src/types/ComponentTypes.ts +39 -0
- package/client/src/types/EdgeCondition.ts +231 -0
- package/client/src/types/INodeProperties.ts +288 -0
- package/client/src/types/NodeTypes.ts +28 -0
- package/client/src/utils/formatters.ts +33 -0
- package/client/src/utils/googleMapsLoader.ts +140 -0
- package/client/src/utils/locationUtils.ts +85 -0
- package/client/src/utils/nodeUtils.ts +31 -0
- package/client/src/utils/workflow.ts +30 -0
- package/client/src/utils/workflowExport.ts +120 -0
- package/client/src/vite-env.d.ts +12 -0
- package/client/tailwind.config.js +60 -0
- package/client/tsconfig.json +25 -0
- package/client/tsconfig.node.json +11 -0
- package/client/vite.config.js +35 -0
- package/docker-compose.prod.yml +107 -0
- package/docker-compose.yml +104 -0
- package/docs-MachinaOs/README.md +85 -0
- package/docs-MachinaOs/deployment/docker.mdx +228 -0
- package/docs-MachinaOs/deployment/production.mdx +345 -0
- package/docs-MachinaOs/docs.json +75 -0
- package/docs-MachinaOs/faq.mdx +309 -0
- package/docs-MachinaOs/favicon.svg +5 -0
- package/docs-MachinaOs/installation.mdx +160 -0
- package/docs-MachinaOs/introduction.mdx +114 -0
- package/docs-MachinaOs/logo/dark.svg +6 -0
- package/docs-MachinaOs/logo/light.svg +6 -0
- package/docs-MachinaOs/nodes/ai-agent.mdx +216 -0
- package/docs-MachinaOs/nodes/ai-models.mdx +240 -0
- package/docs-MachinaOs/nodes/android.mdx +411 -0
- package/docs-MachinaOs/nodes/overview.mdx +181 -0
- package/docs-MachinaOs/nodes/schedulers.mdx +316 -0
- package/docs-MachinaOs/nodes/webhooks.mdx +330 -0
- package/docs-MachinaOs/nodes/whatsapp.mdx +305 -0
- package/docs-MachinaOs/quickstart.mdx +119 -0
- package/docs-MachinaOs/tutorials/ai-agent-workflow.mdx +177 -0
- package/docs-MachinaOs/tutorials/android-automation.mdx +242 -0
- package/docs-MachinaOs/tutorials/first-workflow.mdx +134 -0
- package/docs-MachinaOs/tutorials/whatsapp-automation.mdx +185 -0
- package/nul +0 -0
- package/package.json +70 -0
- package/scripts/build.js +158 -0
- package/scripts/check-ports.ps1 +33 -0
- package/scripts/clean.js +40 -0
- package/scripts/docker.js +93 -0
- package/scripts/kill-port.ps1 +154 -0
- package/scripts/start.js +210 -0
- package/scripts/stop.js +325 -0
- package/server/.dockerignore +44 -0
- package/server/Dockerfile +45 -0
- package/server/constants.py +249 -0
- package/server/core/__init__.py +1 -0
- package/server/core/cache.py +461 -0
- package/server/core/config.py +128 -0
- package/server/core/container.py +99 -0
- package/server/core/database.py +1211 -0
- package/server/core/logging.py +314 -0
- package/server/main.py +289 -0
- package/server/middleware/__init__.py +5 -0
- package/server/middleware/auth.py +89 -0
- package/server/models/__init__.py +1 -0
- package/server/models/auth.py +52 -0
- package/server/models/cache.py +24 -0
- package/server/models/database.py +211 -0
- package/server/models/nodes.py +455 -0
- package/server/package.json +9 -0
- package/server/pyproject.toml +72 -0
- package/server/requirements.txt +83 -0
- package/server/routers/__init__.py +1 -0
- package/server/routers/android.py +294 -0
- package/server/routers/auth.py +203 -0
- package/server/routers/database.py +151 -0
- package/server/routers/maps.py +142 -0
- package/server/routers/nodejs_compat.py +289 -0
- package/server/routers/webhook.py +90 -0
- package/server/routers/websocket.py +2127 -0
- package/server/routers/whatsapp.py +761 -0
- package/server/routers/workflow.py +200 -0
- package/server/services/__init__.py +1 -0
- package/server/services/ai.py +2415 -0
- package/server/services/android/__init__.py +27 -0
- package/server/services/android/broadcaster.py +114 -0
- package/server/services/android/client.py +608 -0
- package/server/services/android/manager.py +78 -0
- package/server/services/android/protocol.py +165 -0
- package/server/services/android_service.py +588 -0
- package/server/services/auth.py +131 -0
- package/server/services/chat_client.py +160 -0
- package/server/services/deployment/__init__.py +12 -0
- package/server/services/deployment/manager.py +706 -0
- package/server/services/deployment/state.py +47 -0
- package/server/services/deployment/triggers.py +275 -0
- package/server/services/event_waiter.py +785 -0
- package/server/services/execution/__init__.py +77 -0
- package/server/services/execution/cache.py +769 -0
- package/server/services/execution/conditions.py +373 -0
- package/server/services/execution/dlq.py +132 -0
- package/server/services/execution/executor.py +1351 -0
- package/server/services/execution/models.py +531 -0
- package/server/services/execution/recovery.py +235 -0
- package/server/services/handlers/__init__.py +126 -0
- package/server/services/handlers/ai.py +355 -0
- package/server/services/handlers/android.py +260 -0
- package/server/services/handlers/code.py +278 -0
- package/server/services/handlers/document.py +598 -0
- package/server/services/handlers/http.py +193 -0
- package/server/services/handlers/polyglot.py +105 -0
- package/server/services/handlers/tools.py +845 -0
- package/server/services/handlers/triggers.py +107 -0
- package/server/services/handlers/utility.py +822 -0
- package/server/services/handlers/whatsapp.py +476 -0
- package/server/services/maps.py +289 -0
- package/server/services/memory_store.py +103 -0
- package/server/services/node_executor.py +375 -0
- package/server/services/parameter_resolver.py +218 -0
- package/server/services/polyglot_client.py +169 -0
- package/server/services/scheduler.py +155 -0
- package/server/services/skill_loader.py +417 -0
- package/server/services/status_broadcaster.py +826 -0
- package/server/services/temporal/__init__.py +23 -0
- package/server/services/temporal/activities.py +344 -0
- package/server/services/temporal/client.py +76 -0
- package/server/services/temporal/executor.py +147 -0
- package/server/services/temporal/worker.py +251 -0
- package/server/services/temporal/workflow.py +355 -0
- package/server/services/temporal/ws_client.py +236 -0
- package/server/services/text.py +111 -0
- package/server/services/user_auth.py +172 -0
- package/server/services/websocket_client.py +29 -0
- package/server/services/workflow.py +597 -0
- package/server/skills/android-skill/SKILL.md +82 -0
- package/server/skills/assistant-personality/SKILL.md +45 -0
- package/server/skills/code-skill/SKILL.md +140 -0
- package/server/skills/http-skill/SKILL.md +161 -0
- package/server/skills/maps-skill/SKILL.md +170 -0
- package/server/skills/memory-skill/SKILL.md +154 -0
- package/server/skills/scheduler-skill/SKILL.md +84 -0
- package/server/skills/whatsapp-skill/SKILL.md +283 -0
- package/server/uv.lock +2916 -0
- package/server/whatsapp-rpc/.dockerignore +30 -0
- package/server/whatsapp-rpc/Dockerfile +44 -0
- package/server/whatsapp-rpc/Dockerfile.web +17 -0
- package/server/whatsapp-rpc/README.md +139 -0
- package/server/whatsapp-rpc/cli.js +95 -0
- package/server/whatsapp-rpc/configs/config.yaml +7 -0
- package/server/whatsapp-rpc/docker-compose.yml +35 -0
- package/server/whatsapp-rpc/docs/API.md +410 -0
- package/server/whatsapp-rpc/go.mod +67 -0
- package/server/whatsapp-rpc/go.sum +203 -0
- package/server/whatsapp-rpc/package.json +30 -0
- package/server/whatsapp-rpc/schema.json +1294 -0
- package/server/whatsapp-rpc/scripts/clean.cjs +66 -0
- package/server/whatsapp-rpc/scripts/cli.js +162 -0
- package/server/whatsapp-rpc/src/go/cmd/server/main.go +91 -0
- package/server/whatsapp-rpc/src/go/config/config.go +49 -0
- package/server/whatsapp-rpc/src/go/rpc/rpc.go +446 -0
- package/server/whatsapp-rpc/src/go/rpc/server.go +112 -0
- package/server/whatsapp-rpc/src/go/whatsapp/history.go +166 -0
- package/server/whatsapp-rpc/src/go/whatsapp/messages.go +390 -0
- package/server/whatsapp-rpc/src/go/whatsapp/service.go +2130 -0
- package/server/whatsapp-rpc/src/go/whatsapp/types.go +261 -0
- package/server/whatsapp-rpc/src/python/pyproject.toml +15 -0
- package/server/whatsapp-rpc/src/python/whatsapp_rpc/__init__.py +4 -0
- package/server/whatsapp-rpc/src/python/whatsapp_rpc/client.py +427 -0
- package/server/whatsapp-rpc/web/app.py +609 -0
- package/server/whatsapp-rpc/web/requirements.txt +6 -0
- package/server/whatsapp-rpc/web/rpc_client.py +427 -0
- package/server/whatsapp-rpc/web/static/openapi.yaml +59 -0
- package/server/whatsapp-rpc/web/templates/base.html +150 -0
- package/server/whatsapp-rpc/web/templates/contacts.html +240 -0
- package/server/whatsapp-rpc/web/templates/dashboard.html +320 -0
- package/server/whatsapp-rpc/web/templates/groups.html +328 -0
- package/server/whatsapp-rpc/web/templates/messages.html +465 -0
- package/server/whatsapp-rpc/web/templates/messaging.html +681 -0
- package/server/whatsapp-rpc/web/templates/send.html +259 -0
- package/server/whatsapp-rpc/web/templates/settings.html +459 -0
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
{% extends "base.html" %}
|
|
2
|
+
{% block title %}Contacts - WhatsApp Controller{% endblock %}
|
|
3
|
+
|
|
4
|
+
{% block content %}
|
|
5
|
+
<div class="space-y-8">
|
|
6
|
+
<!-- Page Header -->
|
|
7
|
+
<div>
|
|
8
|
+
<h1 class="text-3xl font-bold text-gray-900">Contacts</h1>
|
|
9
|
+
<p class="text-gray-600 mt-1">Check if phone numbers are on WhatsApp and view profile pictures</p>
|
|
10
|
+
</div>
|
|
11
|
+
|
|
12
|
+
<!-- Check Numbers Section -->
|
|
13
|
+
<div class="bg-white rounded-lg shadow-md p-6">
|
|
14
|
+
<h2 class="text-xl font-semibold mb-4">Check Phone Numbers</h2>
|
|
15
|
+
<p class="text-gray-600 mb-4">Enter phone numbers (one per line) to check if they are registered on WhatsApp</p>
|
|
16
|
+
|
|
17
|
+
<div class="space-y-4">
|
|
18
|
+
<div>
|
|
19
|
+
<label class="block text-sm font-medium text-gray-700 mb-2">Phone Numbers</label>
|
|
20
|
+
<textarea id="phone-numbers" rows="5"
|
|
21
|
+
placeholder="Enter phone numbers, one per line (without + prefix) e.g. 919876543210 14155551234"
|
|
22
|
+
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-green-500"></textarea>
|
|
23
|
+
</div>
|
|
24
|
+
|
|
25
|
+
<button onclick="checkContacts()" class="px-6 py-2 bg-green-600 text-white rounded-lg hover:bg-green-700 flex items-center">
|
|
26
|
+
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
27
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"></path>
|
|
28
|
+
</svg>
|
|
29
|
+
Check Numbers
|
|
30
|
+
</button>
|
|
31
|
+
</div>
|
|
32
|
+
|
|
33
|
+
<!-- Results -->
|
|
34
|
+
<div id="check-results" class="mt-6 hidden">
|
|
35
|
+
<h3 class="font-semibold text-gray-800 mb-3">Results</h3>
|
|
36
|
+
<div id="results-container" class="space-y-2"></div>
|
|
37
|
+
</div>
|
|
38
|
+
</div>
|
|
39
|
+
|
|
40
|
+
<!-- Profile Picture Section -->
|
|
41
|
+
<div class="bg-white rounded-lg shadow-md p-6">
|
|
42
|
+
<h2 class="text-xl font-semibold mb-4">Profile Picture</h2>
|
|
43
|
+
<p class="text-gray-600 mb-4">View profile picture for a user or group</p>
|
|
44
|
+
|
|
45
|
+
<div class="flex flex-wrap gap-4 items-end">
|
|
46
|
+
<div class="flex-1 min-w-[250px]">
|
|
47
|
+
<label class="block text-sm font-medium text-gray-700 mb-2">JID (Phone or Group)</label>
|
|
48
|
+
<input type="text" id="profile-jid"
|
|
49
|
+
placeholder="919876543210@s.whatsapp.net or group@g.us"
|
|
50
|
+
class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-green-500">
|
|
51
|
+
</div>
|
|
52
|
+
<button onclick="getProfilePicture()" class="px-6 py-2 bg-blue-600 text-white rounded-lg hover:bg-blue-700 flex items-center">
|
|
53
|
+
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
54
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z"></path>
|
|
55
|
+
</svg>
|
|
56
|
+
Get Picture
|
|
57
|
+
</button>
|
|
58
|
+
</div>
|
|
59
|
+
|
|
60
|
+
<!-- Profile Picture Result -->
|
|
61
|
+
<div id="profile-result" class="mt-6 hidden">
|
|
62
|
+
<div id="profile-container" class="flex items-center space-x-4">
|
|
63
|
+
<img id="profile-image" src="" alt="Profile Picture" class="w-24 h-24 rounded-full object-cover border-4 border-gray-200">
|
|
64
|
+
<div>
|
|
65
|
+
<p id="profile-jid-display" class="font-semibold text-gray-800"></p>
|
|
66
|
+
<p id="profile-status" class="text-sm text-gray-600"></p>
|
|
67
|
+
</div>
|
|
68
|
+
</div>
|
|
69
|
+
<div id="profile-no-picture" class="hidden text-gray-500">
|
|
70
|
+
<svg class="w-24 h-24 text-gray-300 mx-auto" fill="currentColor" viewBox="0 0 24 24">
|
|
71
|
+
<path d="M24 20.993V24H0v-2.996A14.977 14.977 0 0112.004 15c4.904 0 9.26 2.354 11.996 5.993zM16.002 8.999a4 4 0 11-8 0 4 4 0 018 0z"></path>
|
|
72
|
+
</svg>
|
|
73
|
+
<p class="text-center mt-2">No profile picture available</p>
|
|
74
|
+
</div>
|
|
75
|
+
</div>
|
|
76
|
+
</div>
|
|
77
|
+
|
|
78
|
+
<!-- Presence Section -->
|
|
79
|
+
<div class="bg-white rounded-lg shadow-md p-6">
|
|
80
|
+
<h2 class="text-xl font-semibold mb-4">Presence Status</h2>
|
|
81
|
+
<p class="text-gray-600 mb-4">Set your online/offline presence status</p>
|
|
82
|
+
|
|
83
|
+
<div class="flex flex-wrap gap-4">
|
|
84
|
+
<button onclick="setPresence('available')" class="px-6 py-2 bg-green-600 text-white rounded-lg hover:bg-green-700 flex items-center">
|
|
85
|
+
<span class="w-3 h-3 bg-green-300 rounded-full mr-2"></span>
|
|
86
|
+
Set Online
|
|
87
|
+
</button>
|
|
88
|
+
<button onclick="setPresence('unavailable')" class="px-6 py-2 bg-gray-600 text-white rounded-lg hover:bg-gray-700 flex items-center">
|
|
89
|
+
<span class="w-3 h-3 bg-gray-400 rounded-full mr-2"></span>
|
|
90
|
+
Set Offline
|
|
91
|
+
</button>
|
|
92
|
+
</div>
|
|
93
|
+
<p id="presence-status" class="mt-3 text-sm text-gray-600"></p>
|
|
94
|
+
</div>
|
|
95
|
+
</div>
|
|
96
|
+
{% endblock %}
|
|
97
|
+
|
|
98
|
+
{% block scripts %}
|
|
99
|
+
<script>
|
|
100
|
+
// Listen for contact check result
|
|
101
|
+
socket.on('contact_check_result', function(data) {
|
|
102
|
+
const resultsSection = document.getElementById('check-results');
|
|
103
|
+
const container = document.getElementById('results-container');
|
|
104
|
+
|
|
105
|
+
resultsSection.classList.remove('hidden');
|
|
106
|
+
|
|
107
|
+
if (!data.success) {
|
|
108
|
+
container.innerHTML = `<div class="text-red-600">Error: ${data.error}</div>`;
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
const results = data.data;
|
|
113
|
+
container.innerHTML = results.map(r => {
|
|
114
|
+
const statusClass = r.is_registered ? 'bg-green-100 border-green-300' : 'bg-red-100 border-red-300';
|
|
115
|
+
const statusIcon = r.is_registered
|
|
116
|
+
? '<svg class="w-5 h-5 text-green-600" fill="currentColor" viewBox="0 0 20 20"><path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clip-rule="evenodd"></path></svg>'
|
|
117
|
+
: '<svg class="w-5 h-5 text-red-600" fill="currentColor" viewBox="0 0 20 20"><path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" clip-rule="evenodd"></path></svg>';
|
|
118
|
+
|
|
119
|
+
let businessBadge = '';
|
|
120
|
+
if (r.is_business) {
|
|
121
|
+
businessBadge = `<span class="ml-2 px-2 py-1 bg-blue-100 text-blue-800 text-xs rounded">Business${r.business_name ? ': ' + escapeHtml(r.business_name) : ''}</span>`;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
return `
|
|
125
|
+
<div class="flex items-center justify-between p-3 border rounded-lg ${statusClass}">
|
|
126
|
+
<div class="flex items-center space-x-3">
|
|
127
|
+
${statusIcon}
|
|
128
|
+
<div>
|
|
129
|
+
<p class="font-medium">${escapeHtml(r.query)}</p>
|
|
130
|
+
${r.is_registered ? `<p class="text-xs text-gray-600">${escapeHtml(r.jid || '')}</p>` : ''}
|
|
131
|
+
</div>
|
|
132
|
+
${businessBadge}
|
|
133
|
+
</div>
|
|
134
|
+
<span class="text-sm font-medium ${r.is_registered ? 'text-green-700' : 'text-red-700'}">
|
|
135
|
+
${r.is_registered ? 'On WhatsApp' : 'Not on WhatsApp'}
|
|
136
|
+
</span>
|
|
137
|
+
</div>
|
|
138
|
+
`;
|
|
139
|
+
}).join('');
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
// Listen for profile picture result
|
|
143
|
+
socket.on('contact_profile_pic_result', function(data) {
|
|
144
|
+
const resultSection = document.getElementById('profile-result');
|
|
145
|
+
const container = document.getElementById('profile-container');
|
|
146
|
+
const noPicture = document.getElementById('profile-no-picture');
|
|
147
|
+
const image = document.getElementById('profile-image');
|
|
148
|
+
const jidDisplay = document.getElementById('profile-jid-display');
|
|
149
|
+
const statusDisplay = document.getElementById('profile-status');
|
|
150
|
+
|
|
151
|
+
resultSection.classList.remove('hidden');
|
|
152
|
+
|
|
153
|
+
if (!data.success) {
|
|
154
|
+
container.classList.add('hidden');
|
|
155
|
+
noPicture.classList.remove('hidden');
|
|
156
|
+
noPicture.innerHTML = `<p class="text-red-600">Error: ${data.error}</p>`;
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
jidDisplay.textContent = data.jid || '';
|
|
161
|
+
|
|
162
|
+
if (data.data.exists) {
|
|
163
|
+
container.classList.remove('hidden');
|
|
164
|
+
noPicture.classList.add('hidden');
|
|
165
|
+
|
|
166
|
+
if (data.data.data) {
|
|
167
|
+
// Base64 data available
|
|
168
|
+
image.src = 'data:image/jpeg;base64,' + data.data.data;
|
|
169
|
+
} else if (data.data.url) {
|
|
170
|
+
// URL available
|
|
171
|
+
image.src = data.data.url;
|
|
172
|
+
}
|
|
173
|
+
statusDisplay.textContent = 'Profile picture found';
|
|
174
|
+
} else {
|
|
175
|
+
container.classList.add('hidden');
|
|
176
|
+
noPicture.classList.remove('hidden');
|
|
177
|
+
noPicture.innerHTML = `
|
|
178
|
+
<svg class="w-24 h-24 text-gray-300 mx-auto" fill="currentColor" viewBox="0 0 24 24">
|
|
179
|
+
<path d="M24 20.993V24H0v-2.996A14.977 14.977 0 0112.004 15c4.904 0 9.26 2.354 11.996 5.993zM16.002 8.999a4 4 0 11-8 0 4 4 0 018 0z"></path>
|
|
180
|
+
</svg>
|
|
181
|
+
<p class="text-center mt-2">No profile picture available</p>
|
|
182
|
+
`;
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
|
|
186
|
+
// Listen for presence result
|
|
187
|
+
socket.on('presence_result', function(data) {
|
|
188
|
+
const status = document.getElementById('presence-status');
|
|
189
|
+
if (data.success) {
|
|
190
|
+
status.textContent = 'Presence updated successfully';
|
|
191
|
+
status.className = 'mt-3 text-sm text-green-600';
|
|
192
|
+
} else {
|
|
193
|
+
status.textContent = 'Failed: ' + data.error;
|
|
194
|
+
status.className = 'mt-3 text-sm text-red-600';
|
|
195
|
+
}
|
|
196
|
+
});
|
|
197
|
+
|
|
198
|
+
function checkContacts() {
|
|
199
|
+
const textarea = document.getElementById('phone-numbers');
|
|
200
|
+
const lines = textarea.value.split('\n').map(l => l.trim()).filter(l => l.length > 0);
|
|
201
|
+
|
|
202
|
+
if (lines.length === 0) {
|
|
203
|
+
alert('Please enter at least one phone number');
|
|
204
|
+
return;
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
socket.emit('contact_check', { phones: lines });
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
function getProfilePicture() {
|
|
211
|
+
const jid = document.getElementById('profile-jid').value.trim();
|
|
212
|
+
|
|
213
|
+
if (!jid) {
|
|
214
|
+
alert('Please enter a JID');
|
|
215
|
+
return;
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
// Auto-append @s.whatsapp.net if no @ present
|
|
219
|
+
let fullJid = jid;
|
|
220
|
+
if (!jid.includes('@')) {
|
|
221
|
+
fullJid = jid + '@s.whatsapp.net';
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
socket.emit('contact_profile_pic', { jid: fullJid, preview: false });
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
function setPresence(status) {
|
|
228
|
+
socket.emit('presence', { status: status });
|
|
229
|
+
document.getElementById('presence-status').textContent = 'Setting presence...';
|
|
230
|
+
document.getElementById('presence-status').className = 'mt-3 text-sm text-blue-600';
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
function escapeHtml(text) {
|
|
234
|
+
if (!text) return '';
|
|
235
|
+
const div = document.createElement('div');
|
|
236
|
+
div.textContent = text;
|
|
237
|
+
return div.innerHTML;
|
|
238
|
+
}
|
|
239
|
+
</script>
|
|
240
|
+
{% endblock %}
|
|
@@ -0,0 +1,320 @@
|
|
|
1
|
+
{% extends "base.html" %}
|
|
2
|
+
|
|
3
|
+
{% block title %}Dashboard - WhatsApp Controller{% endblock %}
|
|
4
|
+
|
|
5
|
+
{% block content %}
|
|
6
|
+
<div class="grid grid-cols-1 lg:grid-cols-3 gap-8">
|
|
7
|
+
|
|
8
|
+
<!-- Status Card -->
|
|
9
|
+
<div class="bg-white rounded-lg shadow-md p-6">
|
|
10
|
+
<h2 class="text-xl font-semibold mb-4 flex items-center">
|
|
11
|
+
<svg class="w-6 h-6 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
12
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"></path>
|
|
13
|
+
</svg>
|
|
14
|
+
Connection Status
|
|
15
|
+
</h2>
|
|
16
|
+
|
|
17
|
+
<div id="status-details" class="space-y-3">
|
|
18
|
+
<div class="flex justify-between items-center">
|
|
19
|
+
<span>Status:</span>
|
|
20
|
+
<span id="connection-status" class="font-medium">
|
|
21
|
+
{{ 'Connected' if status.get('data', {}).get('connected') else 'Disconnected' }}
|
|
22
|
+
</span>
|
|
23
|
+
</div>
|
|
24
|
+
<div class="flex justify-between items-center">
|
|
25
|
+
<span>Session:</span>
|
|
26
|
+
<span id="session-status" class="font-medium">
|
|
27
|
+
{{ 'Active' if status.get('data', {}).get('has_session') else 'None' }}
|
|
28
|
+
</span>
|
|
29
|
+
</div>
|
|
30
|
+
<div class="flex justify-between items-center">
|
|
31
|
+
<span>Last Check:</span>
|
|
32
|
+
<span id="last-check" class="text-sm text-gray-600">
|
|
33
|
+
Never
|
|
34
|
+
</span>
|
|
35
|
+
</div>
|
|
36
|
+
</div>
|
|
37
|
+
|
|
38
|
+
<div class="mt-6 space-y-3">
|
|
39
|
+
<button id="start-btn" onclick="startWhatsApp()"
|
|
40
|
+
class="w-full bg-green-600 text-white py-2 px-4 rounded-lg hover:bg-green-700 transition">
|
|
41
|
+
Start WhatsApp
|
|
42
|
+
</button>
|
|
43
|
+
<button id="restart-btn" onclick="restartWhatsApp()"
|
|
44
|
+
class="w-full bg-orange-600 text-white py-2 px-4 rounded-lg hover:bg-orange-700 transition">
|
|
45
|
+
Restart
|
|
46
|
+
</button>
|
|
47
|
+
</div>
|
|
48
|
+
</div>
|
|
49
|
+
|
|
50
|
+
<!-- QR Code Card -->
|
|
51
|
+
<div class="bg-white rounded-lg shadow-md p-6">
|
|
52
|
+
<h2 class="text-xl font-semibold mb-4 flex items-center">
|
|
53
|
+
<svg class="w-6 h-6 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
54
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197m13.5-9a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z"></path>
|
|
55
|
+
</svg>
|
|
56
|
+
QR Code
|
|
57
|
+
</h2>
|
|
58
|
+
|
|
59
|
+
<div id="qr-container" class="text-center">
|
|
60
|
+
<div class="bg-gray-100 rounded-lg p-8 mb-4">
|
|
61
|
+
<svg class="w-16 h-16 mx-auto text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
62
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197m13.5-9a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z"></path>
|
|
63
|
+
</svg>
|
|
64
|
+
<p class="text-gray-600 mt-2">No QR code available</p>
|
|
65
|
+
</div>
|
|
66
|
+
<p class="text-sm text-gray-600">
|
|
67
|
+
Start WhatsApp service to generate QR code for pairing
|
|
68
|
+
</p>
|
|
69
|
+
</div>
|
|
70
|
+
</div>
|
|
71
|
+
|
|
72
|
+
<!-- Quick Send Card -->
|
|
73
|
+
<div class="bg-white rounded-lg shadow-md p-6">
|
|
74
|
+
<h2 class="text-xl font-semibold mb-4 flex items-center">
|
|
75
|
+
<svg class="w-6 h-6 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
76
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 19l9 2-9-18-9 18 9-2zm0 0v-8"></path>
|
|
77
|
+
</svg>
|
|
78
|
+
Quick Send
|
|
79
|
+
</h2>
|
|
80
|
+
|
|
81
|
+
<form id="quick-send-form" class="space-y-4">
|
|
82
|
+
<div>
|
|
83
|
+
<label class="block text-sm font-medium mb-1">Phone Number</label>
|
|
84
|
+
<input type="text" id="quick-phone" placeholder="1234567890" required
|
|
85
|
+
class="w-full border rounded-lg px-3 py-2 focus:ring-2 focus:ring-green-500 focus:border-transparent">
|
|
86
|
+
</div>
|
|
87
|
+
<div>
|
|
88
|
+
<label class="block text-sm font-medium mb-1">Message</label>
|
|
89
|
+
<textarea id="quick-message" placeholder="Enter your message..." rows="3" required
|
|
90
|
+
class="w-full border rounded-lg px-3 py-2 focus:ring-2 focus:ring-green-500 focus:border-transparent"></textarea>
|
|
91
|
+
</div>
|
|
92
|
+
<button type="submit"
|
|
93
|
+
class="w-full bg-green-600 text-white py-2 px-4 rounded-lg hover:bg-green-700 transition">
|
|
94
|
+
Send Message
|
|
95
|
+
</button>
|
|
96
|
+
</form>
|
|
97
|
+
</div>
|
|
98
|
+
</div>
|
|
99
|
+
|
|
100
|
+
<!-- Activity Feed -->
|
|
101
|
+
<div class="mt-8 bg-white rounded-lg shadow-md p-6">
|
|
102
|
+
<h2 class="text-xl font-semibold mb-4">Activity Feed</h2>
|
|
103
|
+
<div id="activity-feed" class="space-y-2">
|
|
104
|
+
<p class="text-gray-600">No recent activity</p>
|
|
105
|
+
</div>
|
|
106
|
+
</div>
|
|
107
|
+
{% endblock %}
|
|
108
|
+
|
|
109
|
+
{% block scripts %}
|
|
110
|
+
<script>
|
|
111
|
+
// Dashboard specific JavaScript
|
|
112
|
+
let activityFeed = [];
|
|
113
|
+
|
|
114
|
+
function startWhatsApp() {
|
|
115
|
+
const btn = document.getElementById('start-btn');
|
|
116
|
+
btn.disabled = true;
|
|
117
|
+
btn.textContent = 'Starting...';
|
|
118
|
+
|
|
119
|
+
fetch('/api/start', { method: 'POST' })
|
|
120
|
+
.then(response => response.json())
|
|
121
|
+
.then(data => {
|
|
122
|
+
if (data.success) {
|
|
123
|
+
addActivity('success', 'WhatsApp service started successfully');
|
|
124
|
+
// Start loading QR code immediately
|
|
125
|
+
setTimeout(() => loadQRCode(), 1500);
|
|
126
|
+
} else {
|
|
127
|
+
addActivity('error', `Failed to start WhatsApp: ${data.error}`);
|
|
128
|
+
}
|
|
129
|
+
})
|
|
130
|
+
.catch(error => {
|
|
131
|
+
addActivity('error', `Error: ${error.message}`);
|
|
132
|
+
})
|
|
133
|
+
.finally(() => {
|
|
134
|
+
btn.disabled = false;
|
|
135
|
+
btn.textContent = 'Start WhatsApp';
|
|
136
|
+
setTimeout(refreshStatus, 2000);
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
function refreshStatus() {
|
|
141
|
+
fetch('/api/status')
|
|
142
|
+
.then(response => response.json())
|
|
143
|
+
.then(data => {
|
|
144
|
+
if (data.success && data.data) {
|
|
145
|
+
updateStatusDisplay(data.data);
|
|
146
|
+
addActivity('info', 'Status refreshed');
|
|
147
|
+
}
|
|
148
|
+
})
|
|
149
|
+
.catch(error => {
|
|
150
|
+
addActivity('error', `Failed to refresh status: ${error.message}`);
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
function updateStatusDisplay(status) {
|
|
155
|
+
document.getElementById('connection-status').textContent =
|
|
156
|
+
status.connected ? 'Connected' : 'Disconnected';
|
|
157
|
+
document.getElementById('session-status').textContent =
|
|
158
|
+
status.has_session ? 'Active' : 'None';
|
|
159
|
+
document.getElementById('last-check').textContent =
|
|
160
|
+
new Date().toLocaleTimeString();
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
function addActivity(type, message) {
|
|
164
|
+
const timestamp = new Date().toLocaleTimeString();
|
|
165
|
+
activityFeed.unshift({ type, message, timestamp });
|
|
166
|
+
|
|
167
|
+
// Keep only last 10 activities
|
|
168
|
+
if (activityFeed.length > 10) {
|
|
169
|
+
activityFeed = activityFeed.slice(0, 10);
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
renderActivityFeed();
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
function renderActivityFeed() {
|
|
176
|
+
const container = document.getElementById('activity-feed');
|
|
177
|
+
if (activityFeed.length === 0) {
|
|
178
|
+
container.innerHTML = '<p class="text-gray-600">No recent activity</p>';
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
container.innerHTML = activityFeed.map(activity => {
|
|
183
|
+
const colorClass = activity.type === 'success' ? 'text-green-600' :
|
|
184
|
+
activity.type === 'error' ? 'text-red-600' : 'text-blue-600';
|
|
185
|
+
return `
|
|
186
|
+
<div class="flex justify-between items-center py-2 border-b last:border-b-0">
|
|
187
|
+
<span class="${colorClass}">${activity.message}</span>
|
|
188
|
+
<span class="text-sm text-gray-500">${activity.timestamp}</span>
|
|
189
|
+
</div>
|
|
190
|
+
`;
|
|
191
|
+
}).join('');
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
// Quick send form
|
|
195
|
+
document.getElementById('quick-send-form').addEventListener('submit', function(e) {
|
|
196
|
+
e.preventDefault();
|
|
197
|
+
|
|
198
|
+
const phone = document.getElementById('quick-phone').value;
|
|
199
|
+
const message = document.getElementById('quick-message').value;
|
|
200
|
+
|
|
201
|
+
if (!phone || !message) {
|
|
202
|
+
addActivity('error', 'Phone number and message are required');
|
|
203
|
+
return;
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
fetch('/api/send', {
|
|
207
|
+
method: 'POST',
|
|
208
|
+
headers: { 'Content-Type': 'application/json' },
|
|
209
|
+
body: JSON.stringify({ phone, message })
|
|
210
|
+
})
|
|
211
|
+
.then(response => response.json())
|
|
212
|
+
.then(data => {
|
|
213
|
+
if (data.success) {
|
|
214
|
+
addActivity('success', `Message sent to ${phone}`);
|
|
215
|
+
document.getElementById('quick-message').value = '';
|
|
216
|
+
} else {
|
|
217
|
+
addActivity('error', `Failed to send message: ${data.error}`);
|
|
218
|
+
}
|
|
219
|
+
})
|
|
220
|
+
.catch(error => {
|
|
221
|
+
addActivity('error', `Error sending message: ${error.message}`);
|
|
222
|
+
});
|
|
223
|
+
});
|
|
224
|
+
|
|
225
|
+
function loadQRCode(maxAttempts = 10, attemptNum = 1) {
|
|
226
|
+
fetch('/api/qr')
|
|
227
|
+
.then(response => response.json())
|
|
228
|
+
.then(data => {
|
|
229
|
+
const container = document.getElementById('qr-container');
|
|
230
|
+
if (data.success && data.data && data.data.has_qr) {
|
|
231
|
+
// Display base64 image directly (Docker compatible)
|
|
232
|
+
const imgSrc = data.data.image_data
|
|
233
|
+
? `data:image/png;base64,${data.data.image_data}`
|
|
234
|
+
: `/qr/${data.data.filename}`;
|
|
235
|
+
container.innerHTML = `
|
|
236
|
+
<div class="text-center">
|
|
237
|
+
<img src="${imgSrc}" alt="QR Code" class="mx-auto mb-4 border-2 border-gray-300 rounded" style="width: 256px; height: 256px;" />
|
|
238
|
+
<p class="text-sm text-gray-600">Scan this QR code with WhatsApp</p>
|
|
239
|
+
</div>
|
|
240
|
+
`;
|
|
241
|
+
addActivity('info', 'QR code loaded');
|
|
242
|
+
} else {
|
|
243
|
+
// QR not ready yet, poll again if we haven't exceeded max attempts
|
|
244
|
+
if (attemptNum < maxAttempts) {
|
|
245
|
+
container.innerHTML = `
|
|
246
|
+
<div class="bg-gray-100 rounded-lg p-8 mb-4">
|
|
247
|
+
<svg class="w-16 h-16 mx-auto text-gray-400 animate-spin" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
248
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"></path>
|
|
249
|
+
</svg>
|
|
250
|
+
<p class="text-gray-600 mt-2">Waiting for QR code... (${attemptNum}/${maxAttempts})</p>
|
|
251
|
+
</div>
|
|
252
|
+
`;
|
|
253
|
+
setTimeout(() => loadQRCode(maxAttempts, attemptNum + 1), 1000);
|
|
254
|
+
} else {
|
|
255
|
+
container.innerHTML = `
|
|
256
|
+
<div class="bg-gray-100 rounded-lg p-8 mb-4">
|
|
257
|
+
<svg class="w-16 h-16 mx-auto text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
258
|
+
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197m13.5-9a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z"></path>
|
|
259
|
+
</svg>
|
|
260
|
+
<p class="text-gray-600 mt-2">QR code not available</p>
|
|
261
|
+
</div>
|
|
262
|
+
<p class="text-sm text-gray-600">
|
|
263
|
+
Try clicking "Regenerate QR Code" again
|
|
264
|
+
</p>
|
|
265
|
+
`;
|
|
266
|
+
addActivity('error', 'QR code generation timed out');
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
})
|
|
270
|
+
.catch(error => {
|
|
271
|
+
addActivity('error', `Failed to load QR code: ${error.message}`);
|
|
272
|
+
});
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
function restartWhatsApp() {
|
|
276
|
+
const btn = event.target;
|
|
277
|
+
btn.disabled = true;
|
|
278
|
+
btn.textContent = 'Restarting...';
|
|
279
|
+
|
|
280
|
+
fetch('/api/restart', { method: 'POST' })
|
|
281
|
+
.then(response => response.json())
|
|
282
|
+
.then(data => {
|
|
283
|
+
if (data.success) {
|
|
284
|
+
addActivity('success', 'WhatsApp service restarted successfully');
|
|
285
|
+
// Start loading QR code immediately
|
|
286
|
+
setTimeout(() => loadQRCode(), 1500);
|
|
287
|
+
} else {
|
|
288
|
+
addActivity('error', `Failed to restart WhatsApp: ${data.error}`);
|
|
289
|
+
}
|
|
290
|
+
})
|
|
291
|
+
.catch(error => {
|
|
292
|
+
addActivity('error', `Error: ${error.message}`);
|
|
293
|
+
})
|
|
294
|
+
.finally(() => {
|
|
295
|
+
btn.disabled = false;
|
|
296
|
+
btn.textContent = 'Restart';
|
|
297
|
+
setTimeout(refreshStatus, 2000);
|
|
298
|
+
});
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
// Socket.IO events for real-time updates
|
|
302
|
+
socket.on('qr_code', function(data) {
|
|
303
|
+
loadQRCode(); // Reload QR code when event received
|
|
304
|
+
addActivity('info', 'New QR code generated');
|
|
305
|
+
});
|
|
306
|
+
|
|
307
|
+
socket.on('connected', function(data) {
|
|
308
|
+
addActivity('success', 'WhatsApp connected successfully');
|
|
309
|
+
refreshStatus();
|
|
310
|
+
});
|
|
311
|
+
|
|
312
|
+
socket.on('disconnected', function(data) {
|
|
313
|
+
addActivity('warning', 'WhatsApp disconnected');
|
|
314
|
+
refreshStatus();
|
|
315
|
+
});
|
|
316
|
+
|
|
317
|
+
// Initial status check only (QR code must be manually regenerated)
|
|
318
|
+
refreshStatus();
|
|
319
|
+
</script>
|
|
320
|
+
{% endblock %}
|