cbf5c5f834
Ground-up visual redesign across the entire surface (landing, connection setup, chat header, security verification report, file transfer, PWA install/update/offline dialogs). Offline reworked: store-and-forward queue (send while offline → queued, delivered on reconnect), WhatsApp-style per-message delivery status (sending/sent/delivered/not-sent) via delivery receipts, offline buffering for messages to an offline peer, and offline state no longer leaking into the connection indicator. Resilient chunked file transfer with retransmission and auto-save. README + screenshots added.
176 lines
9.0 KiB
React
176 lines
9.0 KiB
React
// "Trusted by our partners" — partner ecosystem section.
|
|
// Translated from the Claude Design component (Partners.dc.html) into the
|
|
// project's React.createElement style: a full-bleed dark band with partner
|
|
// cards plus a dashed "Become a partner" invite card.
|
|
const BecomePartner = () => {
|
|
const [isMobile, setIsMobile] = React.useState(
|
|
typeof window !== 'undefined' && window.matchMedia('(max-width:767px)').matches
|
|
);
|
|
|
|
React.useEffect(() => {
|
|
const mq = window.matchMedia('(max-width:767px)');
|
|
const onChange = () => setIsMobile(mq.matches);
|
|
mq.addEventListener ? mq.addEventListener('change', onChange) : mq.addListener(onChange);
|
|
return () => {
|
|
mq.removeEventListener ? mq.removeEventListener('change', onChange) : mq.removeListener(onChange);
|
|
};
|
|
}, []);
|
|
|
|
const ACCENT = '#f0892a';
|
|
const MONO = "'JetBrains Mono', ui-monospace, SFMono-Regular, Menlo, monospace";
|
|
const SANS = "'Manrope', system-ui, -apple-system, sans-serif";
|
|
|
|
const formUrl = 'https://docs.google.com/forms/d/e/1FAIpQLSc9ijV9PCoyXkus6vEx1OWwvwAsLq8fKS6-H5BmX-c-bvia6w/viewform?usp=dialog';
|
|
|
|
const partners = [
|
|
{
|
|
id: 'aegis',
|
|
name: 'Aegis Investment',
|
|
logo: 'logo/aegis.png',
|
|
logoHeight: '42px',
|
|
url: 'https://aegis-investment.com/',
|
|
desc: 'Capital partner securing confidential financial communications across its portfolio.',
|
|
role: 'Strategic backer',
|
|
delay: '.5s'
|
|
},
|
|
{
|
|
id: 'furi',
|
|
name: 'FuriLabs',
|
|
logo: 'logo/furi.png',
|
|
logoHeight: '54px',
|
|
url: 'https://furilabs.com/',
|
|
desc: 'Privacy-first Linux phones that ship SecureBit as a default secure channel.',
|
|
role: 'Technology partner',
|
|
delay: '.56s'
|
|
}
|
|
];
|
|
|
|
const svg = (inner, size, stroke, sw) =>
|
|
React.createElement('svg', {
|
|
width: size, height: size, viewBox: '0 0 24 24', fill: 'none',
|
|
stroke, strokeWidth: sw, strokeLinecap: 'round', strokeLinejoin: 'round',
|
|
dangerouslySetInnerHTML: { __html: inner }
|
|
});
|
|
|
|
const roleTag = (role) => React.createElement('span', {
|
|
key: 'role',
|
|
style: { fontFamily: MONO, fontSize: '10.5px', fontWeight: 600, color: '#6b6b73', textTransform: 'uppercase', letterSpacing: '1.2px', padding: '6px 11px', borderRadius: '8px', border: '1px solid rgba(255,255,255,0.07)', background: 'rgba(255,255,255,0.025)', whiteSpace: 'nowrap' }
|
|
}, role);
|
|
|
|
const partnerCard = (p) => React.createElement('a', {
|
|
key: p.id,
|
|
href: p.url,
|
|
target: '_blank',
|
|
rel: 'noopener noreferrer',
|
|
style: {
|
|
flex: '1 1 320px', minWidth: isMobile ? 'auto' : '300px',
|
|
borderRadius: '18px', background: '#141416', border: '1px solid rgba(255,255,255,0.06)',
|
|
padding: '30px 30px 26px', display: 'flex', flexDirection: 'column',
|
|
textDecoration: 'none', color: 'inherit',
|
|
transition: 'transform .28s cubic-bezier(.2,.7,.3,1), border-color .28s cubic-bezier(.2,.7,.3,1)',
|
|
animation: `ptUp ${p.delay} cubic-bezier(.2,.7,.3,1)`
|
|
},
|
|
onMouseEnter: (e) => { e.currentTarget.style.transform = 'translateY(-4px)'; e.currentTarget.style.borderColor = 'rgba(255,255,255,0.13)'; },
|
|
onMouseLeave: (e) => { e.currentTarget.style.transform = 'none'; e.currentTarget.style.borderColor = 'rgba(255,255,255,0.06)'; }
|
|
}, [
|
|
React.createElement('div', { key: 'logo', style: { display: 'flex', alignItems: 'center', marginBottom: '30px', height: '54px' } },
|
|
React.createElement('img', {
|
|
src: p.logo, alt: p.name,
|
|
style: { height: p.logoHeight, width: 'auto', maxWidth: '190px', objectFit: 'contain', display: 'block' }
|
|
})
|
|
),
|
|
React.createElement('h3', { key: 'name', style: { margin: '0 0 9px', fontSize: '21px', fontWeight: 800, letterSpacing: '-0.4px', color: '#f4f4f6' } }, p.name),
|
|
React.createElement('p', { key: 'desc', style: { margin: '0 0 22px', fontSize: '14.5px', lineHeight: 1.6, color: '#9a9aa2' } }, p.desc),
|
|
React.createElement('div', { key: 'foot', style: { marginTop: 'auto', paddingTop: '6px', display: 'flex', alignItems: 'center', gap: '12px' } }, [
|
|
roleTag(p.role)
|
|
])
|
|
]);
|
|
|
|
const inviteCard = React.createElement('a', {
|
|
key: 'invite',
|
|
href: formUrl,
|
|
target: '_blank',
|
|
rel: 'noopener noreferrer',
|
|
style: {
|
|
flex: '1 1 320px', minWidth: isMobile ? 'auto' : '300px',
|
|
borderRadius: '18px', background: '#111113', border: '1px dashed rgba(255,255,255,0.12)',
|
|
padding: '30px', display: 'flex', flexDirection: 'column', justifyContent: 'space-between',
|
|
textDecoration: 'none', color: 'inherit',
|
|
transition: 'border-color .28s cubic-bezier(.2,.7,.3,1)',
|
|
animation: 'ptUp .62s cubic-bezier(.2,.7,.3,1)'
|
|
},
|
|
onMouseEnter: (e) => { e.currentTarget.style.borderColor = 'rgba(240,137,42,0.4)'; },
|
|
onMouseLeave: (e) => { e.currentTarget.style.borderColor = 'rgba(255,255,255,0.12)'; }
|
|
}, [
|
|
React.createElement('div', { key: 'top' }, [
|
|
React.createElement('div', {
|
|
key: 'icon',
|
|
style: { width: '48px', height: '48px', borderRadius: '13px', display: 'grid', placeItems: 'center', background: 'rgba(240,137,42,0.12)', border: '1px solid rgba(240,137,42,0.28)', marginBottom: '24px' }
|
|
}, svg('<path d="M16 21v-2a4 4 0 0 0-4-4H6a4 4 0 0 0-4 4v2"/><circle cx="9" cy="7" r="4"/><path d="M19 8v6M22 11h-6"/>', 23, ACCENT, 1.9)),
|
|
React.createElement('h3', { key: 'title', style: { margin: '0 0 8px', fontSize: '21px', fontWeight: 800, letterSpacing: '-0.4px', color: '#f4f4f6' } }, 'Become a partner'),
|
|
React.createElement('p', { key: 'desc', style: { margin: 0, fontSize: '14.5px', lineHeight: 1.6, color: '#8a8a92' } }, "Building privacy hardware or infrastructure? Let's integrate SecureBit.")
|
|
]),
|
|
React.createElement('span', {
|
|
key: 'btn',
|
|
style: {
|
|
marginTop: '26px', width: '100%', display: 'inline-flex', alignItems: 'center', justifyContent: 'center', gap: '10px',
|
|
padding: '15px 20px', borderRadius: '12px', border: 'none', background: ACCENT, color: '#1a0f04',
|
|
fontFamily: SANS, fontSize: '15px', fontWeight: 700, cursor: 'pointer',
|
|
boxShadow: '0 8px 24px rgba(240,137,42,0.28)', boxSizing: 'border-box',
|
|
transition: 'background .2s cubic-bezier(.2,.7,.3,1), transform .2s cubic-bezier(.2,.7,.3,1)'
|
|
}
|
|
}, [
|
|
'Start a conversation',
|
|
svg('<path d="M5 12h14M13 6l6 6-6 6"/>', 17, 'currentColor', 2.2)
|
|
])
|
|
]);
|
|
|
|
const inner = React.createElement('div', {
|
|
key: 'inner',
|
|
style: { maxWidth: '1240px', margin: '0 auto', padding: isMobile ? '0 18px' : '0 40px' }
|
|
}, [
|
|
// Header
|
|
React.createElement('div', { key: 'head', style: { marginBottom: '44px' } }, [
|
|
React.createElement('div', {
|
|
key: 'eyebrow',
|
|
style: { fontFamily: MONO, fontSize: '11px', fontWeight: 600, color: '#6b6b73', textTransform: 'uppercase', letterSpacing: '1.6px', marginBottom: '14px' }
|
|
}, 'Partners & ecosystem'),
|
|
React.createElement('div', {
|
|
key: 'row',
|
|
style: { display: 'flex', alignItems: 'flex-end', justifyContent: 'space-between', gap: '32px', flexWrap: 'wrap' }
|
|
}, [
|
|
React.createElement('h2', {
|
|
key: 'h2',
|
|
style: { margin: 0, fontSize: isMobile ? '30px' : '40px', fontWeight: 800, letterSpacing: '-1.1px', lineHeight: 1.04, color: '#f4f4f6' }
|
|
}, 'Trusted by our partners'),
|
|
React.createElement('p', {
|
|
key: 'sub',
|
|
style: { margin: '0 0 4px', fontSize: '15px', lineHeight: 1.55, color: '#8a8a92', maxWidth: '360px' }
|
|
}, "A small, vetted circle — no pay-to-list logos and no badges we can't stand behind.")
|
|
])
|
|
]),
|
|
|
|
// Cards
|
|
React.createElement('div', {
|
|
key: 'cards',
|
|
style: { display: 'flex', gap: '18px', alignItems: 'stretch', flexWrap: 'wrap' }
|
|
}, [
|
|
...partners.map(partnerCard),
|
|
inviteCard
|
|
])
|
|
]);
|
|
|
|
return React.createElement('section', {
|
|
style: {
|
|
width: '100%', color: '#e8e8eb', fontFamily: SANS,
|
|
padding: isMobile ? '48px 0' : '72px 0',
|
|
background: 'radial-gradient(1100px 640px at 50% -6%, rgba(240,137,42,0.055), transparent 62%), #0f0f11'
|
|
}
|
|
}, [
|
|
React.createElement('style', { key: 'kf', dangerouslySetInnerHTML: { __html: '@keyframes ptUp{from{opacity:0;transform:translateY(12px)}to{opacity:1;transform:translateY(0)}}' } }),
|
|
inner
|
|
]);
|
|
};
|
|
|
|
window.BecomePartner = BecomePartner;
|