File size: 7,568 Bytes
7a38f4f bdec327 7a38f4f 30b1811 7a38f4f 40b20ff 7a38f4f | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 | import React, { useCallback, useEffect, useState } from 'react';
import { Link2, RefreshCw, Loader2 } from 'lucide-react';
import { Button } from '@/components/ui/button';
import { apiFetch } from '@/lib/api';
import { cn } from '@/lib/utils';
export default function LinkedInConnectSettings() {
const [loading, setLoading] = useState(true);
const [busy, setBusy] = useState(false);
const [accounts, setAccounts] = useState([]);
const [defaultRefId, setDefaultRefId] = useState(null);
const load = useCallback(async () => {
setLoading(true);
try {
const r = await apiFetch('/api/unipile/linkedin/campaign-defaults');
const d = r.ok ? await r.json() : {};
setAccounts(d.accounts || []);
setDefaultRefId(d.default_unipile_account_ref_id ?? null);
} catch {
setAccounts([]);
} finally {
setLoading(false);
}
}, []);
useEffect(() => {
load();
}, [load]);
const connectAccount = async () => {
setBusy(true);
try {
const res = await apiFetch('/api/unipile/linkedin/hosted-link', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
label: 'LinkedIn account',
}),
});
const data = await res.json().catch(() => ({}));
if (!res.ok) throw new Error(data.detail || 'Could not start LinkedIn connect');
if (data.url) window.location.href = data.url;
} catch (e) {
alert(e.message || 'Connect failed');
} finally {
setBusy(false);
}
};
return (
<section className="rounded-2xl border border-slate-200 bg-white p-6 shadow-sm">
<div className="flex flex-wrap items-start justify-between gap-3">
<div className="flex items-center gap-2 font-semibold text-slate-800">
<Link2 className="h-5 w-5 text-violet-600" />
Connect LinkedIn
</div>
<Button type="button" variant="outline" size="sm" disabled={loading || busy} onClick={load}>
<RefreshCw className={cn('mr-2 h-4 w-4', loading && 'animate-spin')} />
Refresh
</Button>
</div>
<div className="mt-4 flex flex-col gap-3 md:flex-row md:items-end">
<Button type="button" disabled={busy} onClick={connectAccount} className="bg-slate-900 text-white">
<Link2 className="mr-2 h-4 w-4" />
Connect account
</Button>
</div>
{loading ? (
<div className="mt-6 flex justify-center py-8 text-slate-400">
<Loader2 className="h-8 w-8 animate-spin" />
</div>
) : accounts.length === 0 ? (
<p className="mt-6 text-sm text-slate-500">Connected accounts: 0</p>
) : (
<div className="mt-6 space-y-3">
<p className="text-sm font-medium text-slate-700">
Connected accounts ({accounts.length}) — default for LinkedIn sequence steps
</p>
<ul className="space-y-2">
{accounts.map((a) => {
const name = a.display_name || a.label || 'LinkedIn';
const initials = name
.split(/\s+/)
.slice(0, 2)
.map((s) => s[0] || '')
.join('')
.toUpperCase();
const checked = defaultRefId != null && Number(defaultRefId) === Number(a.id);
return (
<li
key={a.id}
className={cn(
'flex cursor-pointer items-center gap-3 rounded-xl border px-3 py-2 transition',
checked ? 'border-violet-300 bg-violet-50/60' : 'border-slate-200 hover:bg-slate-50'
)}
onClick={() => setDefaultRefId(a.id)}
>
<input
type="radio"
className="h-4 w-4 accent-violet-600"
checked={checked}
onChange={() => setDefaultRefId(a.id)}
/>
{a.avatar_url ? (
<img
src={a.avatar_url}
alt=""
className="h-9 w-9 rounded-full object-cover"
/>
) : (
<span className="flex h-9 w-9 items-center justify-center rounded-full bg-violet-100 text-xs font-semibold text-violet-800">
{initials || 'LI'}
</span>
)}
<div className="min-w-0 flex-1">
<p className="truncate font-medium text-slate-900">{name}</p>
<p className="truncate text-xs text-slate-500">{a.label}</p>
</div>
<span className="text-[10px] font-semibold uppercase text-slate-400">
{a.status || '—'}
</span>
</li>
);
})}
</ul>
<div className="flex flex-wrap items-center gap-3">
<button
type="button"
className="text-xs font-medium text-violet-700 hover:underline"
onClick={async () => {
setBusy(true);
try {
const res = await apiFetch('/api/me/linkedin-prefs', {
method: 'PATCH',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ default_unipile_account_ref_id: 0 }),
});
if (!res.ok) throw new Error('Could not clear');
setDefaultRefId(null);
await load();
} catch (e) {
alert(e.message || 'Failed');
} finally {
setBusy(false);
}
}}
>
Clear default
</button>
</div>
</div>
)}
</section>
);
}
|