{"id":8795,"date":"2026-01-15T08:54:13","date_gmt":"2026-01-15T07:54:13","guid":{"rendered":"https:\/\/innovacore.group\/?post_type=portfolio&p=8795"},"modified":"2026-03-17T15:52:21","modified_gmt":"2026-03-17T13:52:21","slug":"agencia-de-viagens","status":"publish","type":"portfolio","link":"https:\/\/innovacore.group\/pt-pt\/portfolio\/agencia-de-viagens\/","title":{"rendered":"to-travel.agency &#8211; Poder Tecnol\u00f3gico para Profissionais de Viagem"},"content":{"rendered":"\n<p>to-travel.agency is a next-generation tour operator acting as a direct bridge between travel planners, travelers, and local incoming agencies (DMCs). They provide a turnkey technological platform featuring advanced dynamic packaging, democratizing access to enterprise-grade tools. This ecosystem empowers small to mid-sized travel actors to compete with major players by optimizing local expertise and customer service at an affordable cost., with ROI guarantee<\/p>\n\n\n\n<p><\/p>\n","protected":false},"excerpt":{"rendered":"<p>Tecnologia de grande porte e packaging din\u00e2mico para ag\u00eancias pequenas e m\u00e9dias, profissionais de viagem independentes.<\/p>\n","protected":false},"featured_media":2369,"parent":0,"template":"","meta":{"inline_featured_image":false,"company_name":"to-travel.agency - Tech Power for Travel Pros","summary":"Poder Tecnol\u00f3gico para <span class=\"ic-nycd\">Profissionais de Viagem<\/span>","mission":"<p>Democratizar a tecnologia de viagens. A miss\u00e3o deles \u00e9 fornecer aos planeadores independentes e \u00e0s pequenas ag\u00eancias as ferramentas de packaging din\u00e2mico e opera\u00e7\u00f5es necess\u00e1rias para competir com os gigantes do setor, assegurando que o conhecimento local aut\u00eantico permane\u00e7a o valor central da experi\u00eancia de viagem.<\/p>\n","impact":"<p>to-travel.agency est\u00e1 nivelando o campo de jogo no turismo. Ao permitir \u00e0 \"cauda longa\" do mercado aceder a packaging din\u00e2mico sofisticado, eles capacitam especialistas independentes a recuperar quota de mercado de OTAs padronizadas, promovendo uma economia de viagens mais diversa, orientada por especialistas e personalizada.<\/p>\n","company_logo":"https:\/\/innovacore.group\/wp-content\/uploads\/to-travel-logo-innov-innovacore-optimized.jpg","website_url":"","linkedin_url":"","location":"Miami, FL, USA","founded_year":"2022","key_numbers":{"item-0":{"metric_label":"Plataforma Unificada","metric_value":"1"},"item-1":{"metric_label":"% Taxa de Crescimento Anual","metric_value":"40"},"item-2":{"metric_label":"% EBITDA Esperada","metric_value":"30"}},"hiring_status":"discreta","sector":"outro","company_type":"startup","audience":["b2b_a_b2c","b2b2c"],"geo_scope":["global"],"business_model":["mercado","servi\u00e7os","assinatura"],"funding_stage":"pr\u00e9_seed","partnership_type":"cria\u00e7\u00e3o","logo_animation":"<div id=\"totravel-particle-root\">\r\n    <canvas id=\"totravelCanvas\"><\/canvas>\r\n<\/div>\r\n\r\n<style>\r\n    #totravel-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;\r\n        overflow: hidden;\r\n        cursor: crosshair;\r\n    }\r\n<\/style>\r\n\r\n<script>\r\n(function() {\r\n    const canvas = document.getElementById('totravelCanvas');\r\n    const ctx = canvas.getContext('2d', { willReadFrequently: true });\r\n    const root = document.getElementById('totravel-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 Innovacore\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        const cx = width \/ 2;\r\n        const cy = height \/ 2;\r\n        const isMobile = width < 600;\r\n\r\n        \/\/ --- 1. CONFIGURA\u00c7\u00c3O ---\r\n        ctx.fillStyle = 'white';\r\n        ctx.strokeStyle = 'white';\r\n\r\n        \/\/ Dimens\u00f5es relativas\r\n        const iconSize = isMobile ? 50 : 80;   \/\/ Taille du globe\r\n        const fontSize = isMobile ? 24 : 45;   \/\/ Taille du texte principal\r\n        const fontSizeSub = isMobile ? 12 : 18; \/\/ Taille tagline\r\n        const gap = isMobile ? 10 : 20;        \/\/ Espa\u00e7o Icone <-> Texto\r\n\r\n        \/\/ Police de base\r\n        const fontNormal = `normal ${fontSize}px Arial, Helvetica, sans-serif`;\r\n        const fontBold = `bold ${fontSize}px Arial, Helvetica, sans-serif`;\r\n        const fontSub = `normal ${fontSizeSub}px Arial, Helvetica, sans-serif`;\r\n\r\n        \/\/ Calcul des largeurs de texte pour l'alignement\r\n        ctx.font = fontNormal;\r\n        const wTo = ctx.measureText('to-').width;\r\n        const wAgency = ctx.measureText('.agency').width;\r\n        \r\n        ctx.font = fontBold;\r\n        const wTravel = ctx.measureText('travel').width;\r\n        \r\n        const totalTextWidth = wTo + wTravel + wAgency;\r\n        \r\n        \/\/ Largeur totale approximative (Globe + Gap + Textes)\r\n        \/\/ Le globe est un peu en fond, on le compte partiellement\r\n        const totalContentWidth = (iconSize * 0.8) + gap + totalTextWidth;\r\n        const startX = cx - (totalContentWidth \/ 2);\r\n\r\n\r\n        \/\/ --- 2. DESCRI\u00c7\u00c3O DO GLOBE ---\r\n        const ix = startX + (iconSize \/ 2);\r\n        const iy = cy; \/\/ Centrado verticalmente com o texto\r\n        \r\n        \/\/ Cercle ext\u00e9rieur (fin)\r\n        ctx.lineWidth = 2;\r\n        ctx.beginPath();\r\n        ctx.arc(ix, iy, iconSize \/ 2, 0, Math.PI * 2);\r\n        ctx.stroke();\r\n\r\n        \/\/ Continentes (Formes simplifi\u00e9es para sugerir a mapa)\r\n        ctx.fillStyle = 'white';\r\n        ctx.beginPath();\r\n        \r\n        \/\/ Am\u00e9rique du Nord (Blob haut gauche)\r\n        ctx.moveTo(ix - iconSize*0.2, iy - iconSize*0.3);\r\n        ctx.quadraticCurveTo(ix - iconSize*0.4, iy - iconSize*0.1, ix - iconSize*0.3, iy + iconSize*0.1);\r\n        ctx.lineTo(ix - iconSize*0.1, iy);\r\n        \r\n        \/\/ Am\u00e9rique du Sud (Blob bas gauche)\r\n        ctx.moveTo(ix - iconSize*0.25, iy + iconSize*0.15);\r\n        ctx.quadraticCurveTo(ix - iconSize*0.15, iy + iconSize*0.35, ix - iconSize*0.05, iy + iconSize*0.1);\r\n        \r\n        \/\/ Europe\/Afrique (Blob droit)\r\n        ctx.moveTo(ix + iconSize*0.1, iy - iconSize*0.25);\r\n        ctx.quadraticCurveTo(ix + iconSize*0.3, iy - iconSize*0.1, ix + iconSize*0.2, iy + iconSize*0.2);\r\n        ctx.quadraticCurveTo(ix + iconSize*0.1, iy + iconSize*0.3, ix, iy + iconSize*0.1);\r\n\r\n        ctx.fill();\r\n\r\n\r\n        \/\/ --- 3. DESCRI\u00c7\u00c3O DO TEXTO ---\r\n        \/\/ Desloca o texto para combinar com o globo ou ficar ao lado\r\n        const textStartX = startX + (iconSize * 0.8) + gap;\r\n        const textY = cy; \/\/ Linha de base centrada\r\n        \r\n        ctx.textAlign = 'left';\r\n        ctx.textBaseline = 'middle';\r\n\r\n        \/\/ \"to-\"\r\n        ctx.font = fontNormal;\r\n        ctx.fillText('to-', textStartX, textY);\r\n\r\n        \/\/ \"travel\" (Gras)\r\n        ctx.font = fontBold;\r\n        ctx.fillText('travel', textStartX + wTo, textY);\r\n\r\n        \/\/ \".agency\"\r\n        ctx.font = fontNormal;\r\n        ctx.fillText('.agency', textStartX + wTo + wTravel, textY);\r\n\r\n        \/\/ Tagline \"travel agent helper\"\r\n        \/\/ Alinhada \u00e0 direita ou centrada sob o texto principal\r\n        ctx.font = fontSub;\r\n        ctx.textAlign = 'right'; \/\/ Alinhamento \u00e0 direita para ficar bonito sob \".agency\"\r\n        ctx.fillText('travel agent helper', textStartX + totalTextWidth, textY + fontSize * 0.8);\r\n\r\n\r\n        \/\/ --- 4. SCAN DE PIXELS ---\r\n        const density = 4;\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                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,\r\n                    });\r\n                }\r\n            }\r\n        }\r\n        \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        \r\n        \/\/ --- F\u00cdSICA LENTA & SUAVE ---\r\n        time += 0.02;\r\n\r\n        ctx.fillStyle = particleColor;\r\n        \r\n        particles.forEach(p => {\r\n            \/\/ Onda\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            \/\/ Intera\u00e7\u00e3o\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 * 8;\r\n                repelY = -Math.sin(angle) * force * 8;\r\n            }\r\n\r\n            const targetX = p.originX + waveX;\r\n            const targetY = p.originY + waveY;\r\n\r\n            \/\/ Retorno suave\r\n            const ax = (targetX - p.x) * 0.03;\r\n            const ay = (targetY - p.y) * 0.03;\r\n\r\n            p.vx += ax;\r\n            p.vy += ay;\r\n            p.vx += repelX * 0.3;\r\n            p.vy += repelY * 0.3;\r\n            \r\n            \/\/ Fric\u00e7\u00e3o\r\n            p.vx *= 0.90; \r\n            p.vy *= 0.90;\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    setTimeout(init, 100);\r\n    draw();\r\n})();\r\n<\/script>"},"pt":[],"pipeline":[],"class_list":["post-8795","portfolio","type-portfolio","status-publish","has-post-thumbnail","hentry"],"_links":{"self":[{"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/portfolio\/8795","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/portfolio"}],"about":[{"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/types\/portfolio"}],"version-history":[{"count":1,"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/portfolio\/8795\/revisions"}],"predecessor-version":[{"id":8796,"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/portfolio\/8795\/revisions\/8796"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/media\/2369"}],"wp:attachment":[{"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/media?parent=8795"}],"wp:term":[{"taxonomy":"pt","embeddable":true,"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/pt?post=8795"},{"taxonomy":"pipeline","embeddable":true,"href":"https:\/\/innovacore.group\/pt-pt\/wp-json\/wp\/v2\/pipeline?post=8795"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}