{"id":3239,"date":"2026-01-14T11:58:28","date_gmt":"2026-01-14T11:58:28","guid":{"rendered":"https:\/\/test.innovacore.group\/?post_type=portfolio&#038;p=3239"},"modified":"2026-01-15T11:59:16","modified_gmt":"2026-01-15T11:59:16","slug":"source-agency","status":"publish","type":"portfolio","link":"https:\/\/test.innovacore.group\/fr\/portfolio\/source-agency\/","title":{"rendered":"Source Agency \u2013 The Only Results-Driven Marketing Support"},"content":{"rendered":"\n<p class=\"wp-block-paragraph\">Source Agency (Source Group Marketing) is a global digital and creative marketing partner helping ambitious brands enter new markets and scale internationally. We unite strategy, branding, web, SEO\/SEA, social and paid media with AI-assisted insights and automation to test faster, personalize journeys and convert more. The result: clear positioning, qualified leads and sustainable growth.<\/p>\n","protected":false},"excerpt":{"rendered":"<p>AI-driven growth marketing for brands scaling internationally.<\/p>\n","protected":false},"featured_media":2369,"parent":0,"template":"","meta":{"inline_featured_image":false,"company_name":"Source Agency","summary":"The Only <span class=\"ic-nycd\">Results-Driven<\/span> Marketing Support","mission":"<p>Mission: Help visionary entrepreneurs and teams win globally with bespoke go-to-market and growth strategies. We combine human creativity with AI-assisted analysis, rapid experimentation and automation to build visibility, generate demand and improve profitability.<\/p>\n","impact":"<p>Impact: Faster decisions and smarter campaigns through data and AI-assisted testing. Better traffic quality, higher conversion and stronger brand presence across markets. Automated nurturing and CRM-ready workflows turn prospects into customers and support long-term, measurable growth.<\/p>\n","company_logo":"https:\/\/test.innovacore.group\/wp-content\/uploads\/source-logo-core-innovacore-optimized.jpg","website_url":"https:\/\/sourcegroup.marketing","linkedin_url":"https:\/\/www.linkedin.com\/company\/sourceagency\/","location":"Miami","founded_year":"2020","key_numbers":{"item-0":{"metric_label":"+M$ generated","metric_value":"300"},"item-1":{"metric_label":"+ markets activated","metric_value":"20"},"item-2":{"metric_label":"+% growth rate","metric_value":"30"}},"hiring_status":"hiring","sector":"other","company_type":"sme","audience":["b2b","b2b2c"],"geo_scope":["global"],"business_model":["services"],"funding_stage":"undisclosed","partnership_type":"creation","logo_animation":"<div id=\"source-particle-root\">\r\n    <canvas id=\"sourceCanvas\"><\/canvas>\r\n<\/div>\r\n<\/html>\r\n<style>\r\n    #source-particle-root {\r\n        width: 100%;\r\n        height: 300px; \/* Hauteur fixe respect\u00e9e *\/\r\n        display: flex;\r\n        justify-content: center;\r\n        align-items: center;\r\n        background: transparent; \/* Fond transparent pour s'int\u00e9grer \u00e0 ton site *\/\r\n        overflow: hidden;\r\n        cursor: crosshair; \/* Invite \u00e0 l'interaction *\/\r\n    }\r\n<\/style>\r\n\r\n<script>\r\n(function() {\r\n    const canvas = document.getElementById('sourceCanvas');\r\n    const ctx = canvas.getContext('2d', { willReadFrequently: true });\r\n    const root = document.getElementById('source-particle-root');\r\n    \r\n    let width, height;\r\n    let particles = [];\r\n    let mouse = { x: -1000, y: -1000 };\r\n    const particleColor = '#f5f5f5'; \/\/ Gris clair pour ressortir sur ton fond sombre\r\n    let time = 0;\r\n\r\n    function init() {\r\n        width = root.offsetWidth;\r\n        height = 300;\r\n        \r\n        const dpr = window.devicePixelRatio || 1;\r\n        canvas.width = width * dpr;\r\n        canvas.height = height * dpr;\r\n        canvas.style.width = width + 'px';\r\n        canvas.style.height = height + 'px';\r\n        ctx.scale(dpr, dpr);\r\n\r\n        createLogoParticles();\r\n    }\r\n\r\n    function createLogoParticles() {\r\n        particles = [];\r\n        \r\n        \/\/ 1. Configuration du texte \"source.\"\r\n        \/\/ Taille adaptative : plus petit sur mobile (60px) vs desktop (130px)\r\n        const fontSize = width < 600 ? 60 : 130; \r\n        \r\n        ctx.font = `bold ${fontSize}px \"Times New Roman\", Times, serif`;\r\n        ctx.fillStyle = 'white'; \/\/ Couleur technique pour le scan (invisible \u00e0 la fin)\r\n        ctx.textAlign = 'center';\r\n        ctx.textBaseline = 'middle';\r\n        \r\n        \/\/ Dessin au centre exact\r\n        ctx.fillText('source.', width \/ 2, height \/ 2);\r\n\r\n        \/\/ 2. Scan des pixels\r\n        const density = 4; \/\/ \u00c9chantillonnage (4 = bon \u00e9quilibre perf\/qualit\u00e9)\r\n        const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height).data;\r\n        const dataWidth = canvas.width; \r\n\r\n        for (let y = 0; y < canvas.height; y += density) {\r\n            for (let x = 0; x < canvas.width; x += density) {\r\n                \/\/ Si le pixel n'est pas transparent\r\n                if (imageData[(y * dataWidth + x) * 4 + 3] > 128) {\r\n                    const dpr = window.devicePixelRatio || 1;\r\n                    particles.push({\r\n                        x: x \/ dpr,\r\n                        y: y \/ dpr,\r\n                        originX: x \/ dpr,\r\n                        originY: y \/ dpr,\r\n                        vx: 0, \r\n                        vy: 0,\r\n                        size: Math.random() * 1.5 + 0.5, \/\/ Taille variable naturelle\r\n                        force: Math.random() \/\/ Variation individuelle\r\n                    });\r\n                }\r\n            }\r\n        }\r\n        \r\n        \/\/ Nettoyage imm\u00e9diat du texte statique pour ne laisser que les particules\r\n        ctx.clearRect(0, 0, width, height);\r\n    }\r\n\r\n    function draw() {\r\n        ctx.clearRect(0, 0, width, height);\r\n        time += 0.05;\r\n\r\n        ctx.fillStyle = particleColor;\r\n        \r\n        particles.forEach(p => {\r\n            \/\/ --- 1. Mouvement naturel (Respiration) ---\r\n            const waveX = Math.sin(time * 0.5 + p.y * 0.05) * 2;\r\n            const waveY = Math.cos(time * 0.3 + p.x * 0.05) * 2;\r\n            \r\n            \/\/ --- 2. Interaction Souris ---\r\n            const dx = mouse.x - p.x;\r\n            const dy = mouse.y - p.y;\r\n            const dist = Math.sqrt(dx*dx + dy*dy);\r\n            const radius = 80;\r\n            \r\n            let repelX = 0;\r\n            let repelY = 0;\r\n\r\n            if (dist < radius) {\r\n                const force = (radius - dist) \/ radius;\r\n                const angle = Math.atan2(dy, dx);\r\n                repelX = -Math.cos(angle) * force * 20;\r\n                repelY = -Math.sin(angle) * force * 20;\r\n            }\r\n\r\n            \/\/ --- 3. Physique (Retour \u00e9lastique) ---\r\n            const targetX = p.originX + waveX;\r\n            const targetY = p.originY + waveY;\r\n\r\n            const ax = (targetX - p.x) * 0.08;\r\n            const ay = (targetY - p.y) * 0.08;\r\n\r\n            p.vx += ax;\r\n            p.vy += ay;\r\n            p.vx += repelX * 0.5;\r\n            p.vy += repelY * 0.5;\r\n            p.vx *= 0.85; \/\/ Friction\r\n            p.vy *= 0.85;\r\n\r\n            p.x += p.vx;\r\n            p.y += p.vy;\r\n\r\n            ctx.beginPath();\r\n            ctx.arc(p.x, p.y, p.size, 0, Math.PI * 2);\r\n            ctx.fill();\r\n        });\r\n\r\n        requestAnimationFrame(draw);\r\n    }\r\n\r\n    const updateMouse = (e) => {\r\n        const r = canvas.getBoundingClientRect();\r\n        const cx = e.touches ? e.touches[0].clientX : e.clientX;\r\n        const cy = e.touches ? e.touches[0].clientY : e.clientY;\r\n        mouse.x = cx - r.left;\r\n        mouse.y = cy - r.top;\r\n    };\r\n\r\n    window.addEventListener('resize', init);\r\n    canvas.addEventListener('mousemove', updateMouse);\r\n    canvas.addEventListener('touchmove', updateMouse, {passive: true});\r\n    canvas.addEventListener('mouseleave', () => { mouse.x = -1000; mouse.y = -1000; });\r\n    canvas.addEventListener('touchend', () => { mouse.x = -1000; mouse.y = -1000; });\r\n\r\n    \/\/ Petit d\u00e9lai pour assurer le chargement de la police\r\n    setTimeout(init, 50);\r\n    draw();\r\n})();\r\n<\/script>"},"pt":[],"pipeline":[33,57],"class_list":["post-3239","portfolio","type-portfolio","status-publish","has-post-thumbnail","hentry","pipeline-core","pipeline-marketing"],"_links":{"self":[{"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/portfolio\/3239","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/portfolio"}],"about":[{"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/types\/portfolio"}],"version-history":[{"count":6,"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/portfolio\/3239\/revisions"}],"predecessor-version":[{"id":3463,"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/portfolio\/3239\/revisions\/3463"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/media\/2369"}],"wp:attachment":[{"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/media?parent=3239"}],"wp:term":[{"taxonomy":"pt","embeddable":true,"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/pt?post=3239"},{"taxonomy":"pipeline","embeddable":true,"href":"https:\/\/test.innovacore.group\/fr\/wp-json\/wp\/v2\/pipeline?post=3239"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}