Files
cv/search_analyzer.js
2026-06-05 22:29:04 +03:00

329 lines
13 KiB
JavaScript

// search_analyzer.js — LinkedIn People Search Investor Analyzer
// Operates on linkedin.com/search/results/people/*
(function () {
'use strict';
console.log('[LJA] Search Analyzer Script Loaded');
// Prevent double injection
if (window.__linkedinSearchAnalyzerLoaded) return;
window.__linkedinSearchAnalyzerLoaded = true;
// ─── Utility: get stored settings ────────────────────────────────────────
function getSettings() {
return new Promise(resolve => {
if (!chrome || !chrome.storage) {
resolve({});
return;
}
chrome.storage.sync.get(['apiKey', 'language', 'userProfile'], (syncData) => {
if (syncData && syncData.apiKey) {
resolve(syncData);
} else {
chrome.storage.local.get(['apiKey', 'language', 'userProfile'], (localData) => {
resolve(localData || {});
});
}
});
});
}
// ─── Find Result Container ───────────────────────────────────────────────
function getSearchResultsContainer() {
return document.querySelector('.search-results-container, .reusable-search__entity-result-list, ul.reusable-search__entity-result-list, .search-results__list');
}
// ─── Extract Person Data ─────────────────────────────────────────────────
function extractPersonData(cardEl) {
const data = {
name: '',
headline: '',
location: '',
summary: ''
};
// 1. Extract Name (Usually inside a link with /in/)
const nameEl = cardEl.querySelector('.entity-result__title-text, .search-result__title, span[dir="ltr"]');
if (nameEl) {
data.name = nameEl.innerText.trim().split('\n')[0];
} else {
const profileLinks = Array.from(cardEl.querySelectorAll('a[href*="/in/"]')).filter(a => a.innerText.trim().length > 0);
if (profileLinks.length > 0) {
data.name = profileLinks[0].innerText.trim().split('\n')[0];
}
}
// 2. Extract Headline
const headlineEl = cardEl.querySelector('.entity-result__primary-subtitle, .search-result__truncate, .linked-area');
if (headlineEl) {
data.headline = headlineEl.innerText.trim();
}
// 3. Extract Location
const locationEl = cardEl.querySelector('.entity-result__secondary-subtitle, .search-result__info');
if (locationEl) {
data.location = locationEl.innerText.trim();
}
// 4. Extract Summary
const summaryEl = cardEl.querySelector('.entity-result__summary, .search-result__snippets');
if (summaryEl) {
data.summary = summaryEl.innerText.trim();
}
// Clean up "View Profile" texts
data.name = data.name.replace(/View .* profile/gi, '').trim();
return data;
}
// ─── Inject UI into Card ─────────────────────────────────────────────────
function injectScanButton(cardEl) {
if (cardEl.querySelector('.lja-scan-person-btn') || cardEl.querySelector('.lja-investor-result')) return;
const btn = document.createElement('button');
btn.className = 'lja-scan-person-btn';
btn.innerHTML = '🔍 Scan Investor';
// Add some basic styling just in case CSS fails to load or apply correctly
btn.style.margin = '10px 0';
btn.style.padding = '5px 15px';
btn.style.cursor = 'pointer';
btn.style.backgroundColor = '#6C63FF';
btn.style.color = '#fff';
btn.style.border = 'none';
btn.style.borderRadius = '5px';
btn.style.fontWeight = 'bold';
// Create a container for the result
const resultContainer = document.createElement('div');
resultContainer.className = 'lja-result-wrapper';
btn.addEventListener('click', async (e) => {
e.stopPropagation();
e.preventDefault();
await scanPerson(cardEl, btn, resultContainer);
});
// Try to append to actions area, if not just append at the end of the card
const actionArea = cardEl.querySelector('.entity-result__actions, .search-result__actions');
if (actionArea) {
actionArea.prepend(btn);
} else {
cardEl.appendChild(btn);
}
cardEl.appendChild(resultContainer);
console.log('[LJA] Injected button for a profile:', extractPersonData(cardEl).name);
}
// ─── Scan a Single Person ────────────────────────────────────────────────
async function scanPerson(cardEl, btnEl, resultContainer) {
const settings = await getSettings();
if (!settings || !settings.apiKey) {
alert('Please set your Gemini API key in the extension popup first.');
return;
}
const data = extractPersonData(cardEl);
if (!data.name && !data.headline) {
alert('Could not extract person details.');
return;
}
btnEl.disabled = true;
btnEl.innerHTML = '<span class="lja-spinner"></span> Scanning...';
const prompt = `أنت مستشار استثماري خبير في الشركات الناشئة (Startups) في الشرق الأوسط ومصر.
المستخدم يبحث عن مستثمرين (Angel Investors) أو أشخاص يمكنهم توصيله بمستثمرين لتطبيقاته "انطلق" (Intaleq) و "تريبز" (Tripz) وهي تطبيقات نقل ذكي (Ride-hailing) في مصر والشرق الأوسط.
يرجى تقييم هذا الشخص بناءً على المعلومات التالية المستخرجة من لينكد إن:
الاسم: ${data.name}
المسمى الوظيفي: ${data.headline}
الموقع: ${data.location}
نبذة: ${data.summary}
هل هذا الشخص مناسب للتواصل معه وعرض الاستثمار أو طلب توصية بمستثمرين؟
إذا كان الشخص مستثمرًا فعليًا أو يعمل في صندوق استثماري (VC) أو لديه شبكة علاقات قوية في ريادة الأعمال أو يدعم الشركات الناشئة، أعطه العلامة الخضراء (green).
إذا كان الشخص لا يبدو مرتبطًا بالاستثمار (مثلاً مجرد موظف عادي، أو يبحث عن عمل، أو في مجال بعيد جداً عن دعم الشركات)، أعطه العلامة الحمراء (red) لكي لا يضيع المستخدم وقته معه.
يجب أن يكون الرد بصيغة JSON فقط بهذا الشكل:
{
"status": "green" أو "red",
"reason": "سبب قصير جداً باللغة العربية يشرح لماذا (سطر واحد)"
}
لا تقم بإضافة أي نص آخر أو Markdown.`;
try {
const response = await chrome.runtime.sendMessage({
type: 'GEMINI_REQUEST',
payload: {
apiKey: settings.apiKey,
action: 'generateText',
prompt: prompt
}
});
if (!response.success) {
throw new Error(response.error || 'Unknown error');
}
let rawText = response.data.text || response.data;
rawText = rawText.replace(/```json/gi, '').replace(/```/g, '').trim();
let resultData;
try {
resultData = JSON.parse(rawText);
} catch (parseError) {
throw new Error('Failed to parse AI response. Raw: ' + rawText);
}
const isGreen = resultData.status === 'green';
const badgeIcon = isGreen ? '✅' : '❌';
const badgeText = isGreen ? 'تواصل معه' : 'تجاهله';
const colorClass = isGreen ? 'green' : 'red';
resultContainer.innerHTML = `
<div class="lja-investor-result ${colorClass}" dir="rtl" style="margin-top: 10px; padding: 10px; border-radius: 8px; font-weight: bold; font-family: system-ui; background-color: ${isGreen ? '#e6ffe6' : '#ffe6e6'}; color: ${isGreen ? '#006600' : '#cc0000'}; border: 1px solid ${isGreen ? '#00cc00' : '#ff0000'};">
<div class="lja-investor-badge">
<span>${badgeIcon}</span> ${badgeText}
</div>
<div class="lja-investor-reason" style="margin-top: 5px; font-weight: normal; font-size: 14px;">
${resultData.reason}
</div>
</div>
`;
btnEl.style.display = 'none';
} catch (e) {
console.error('[LJA Search]', e);
resultContainer.innerHTML = `<div class="lja-investor-result red" style="color:red; font-weight:bold;">❌ Error: ${e.message}</div>`;
btnEl.disabled = false;
btnEl.innerHTML = '🔍 Scan Investor';
}
}
// ─── Scan All Feature ────────────────────────────────────────────────────
function injectScanAllButton() {
if (document.querySelector('.lja-scan-list-btn')) return;
let container = getSearchResultsContainer();
const btn = document.createElement('button');
btn.className = 'lja-scan-list-btn';
btn.innerHTML = '✨ Scan All Investors';
btn.title = 'Scan all loaded profiles on this page';
// Robust styling for the Scan All button
btn.style.margin = '20px auto';
btn.style.display = 'block';
btn.style.padding = '10px 20px';
btn.style.cursor = 'pointer';
btn.style.backgroundColor = '#6C63FF';
btn.style.color = '#fff';
btn.style.border = 'none';
btn.style.borderRadius = '8px';
btn.style.fontWeight = 'bold';
btn.style.fontSize = '16px';
btn.style.boxShadow = '0 4px 6px rgba(0,0,0,0.1)';
btn.addEventListener('click', async () => {
// Find all cards using our robust selector
const allLis = Array.from(document.querySelectorAll('li'));
const cards = allLis.filter(li => li.querySelector('a[href*="/in/"]'));
if (cards.length === 0) {
alert('No profiles found to scan.');
return;
}
btn.disabled = true;
btn.innerHTML = '<span class="lja-spinner"></span> Scanning profiles...';
// Scan sequentially
for (const card of cards) {
const scanBtn = card.querySelector('.lja-scan-person-btn');
if (scanBtn && scanBtn.style.display !== 'none') {
card.scrollIntoView({ behavior: 'smooth', block: 'center' });
await new Promise(r => setTimeout(r, 500));
scanBtn.click();
while (scanBtn.disabled && scanBtn.style.display !== 'none') {
await new Promise(r => setTimeout(r, 500));
}
}
}
btn.disabled = false;
btn.innerHTML = '✨ Scan Complete!';
setTimeout(() => { btn.innerHTML = '✨ Scan All Investors'; }, 3000);
});
if (container && container.parentNode) {
container.parentNode.insertBefore(btn, container);
} else {
// Fallback: Floating button bottom right
btn.style.position = 'fixed';
btn.style.bottom = '20px';
btn.style.right = '20px';
btn.style.zIndex = '99999';
btn.style.margin = '0';
document.body.appendChild(btn);
}
console.log('[LJA] Injected Scan All button');
}
// ─── Process Page ────────────────────────────────────────────────────────
function processPage() {
if (!window.location.href.includes('linkedin.com/search/results/')) return;
// Robustly find any <li> that contains a link to a LinkedIn profile
const allLis = Array.from(document.querySelectorAll('li'));
const cards = allLis.filter(li => li.querySelector('a[href*="/in/"]'));
console.log('[LJA] processPage found cards:', cards.length);
if (cards.length > 0) {
injectScanAllButton();
cards.forEach(injectScanButton);
}
}
// ─── MutationObserver: watch for new results (pagination/filters) ────────
let observerTimer;
const observer = new MutationObserver(() => {
if (observerTimer) return;
observerTimer = setTimeout(() => {
processPage();
observerTimer = null;
}, 1500);
});
// ─── Initialize ──────────────────────────────────────────────────────────
function init() {
console.log('[LJA] Initializing Search Analyzer...');
processPage();
observer.observe(document.body, { childList: true, subtree: true });
}
// Handle SPA navigation by checking URL changes
let lastUrl = window.location.href;
setInterval(() => {
if (window.location.href !== lastUrl) {
lastUrl = window.location.href;
if (lastUrl.includes('linkedin.com/search/results/')) {
console.log('[LJA] SPA Navigation detected, re-processing...');
setTimeout(processPage, 2000);
}
}
}, 1000);
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', init);
} else {
init();
setTimeout(init, 3000); // Fallback for delayed renders
}
})();