from dotenv import load_dotenv import streamlit as st import os import google.generativeai as genai import random from streamlit import session_state as state from formulas import headline_formulas from angles import angles # Cargar las variables de entorno load_dotenv() # Configurar la API de Google genai.configure(api_key=os.getenv("GOOGLE_API_KEY")) # Fórmulas con ejemplos y explicaciones # headline_formulas dictionary has been moved to formulas/headline_formulas.py def generate_headlines(number_of_headlines, target_audience, product, temperature, selected_formula, selected_angle): # Crear la configuración del modelo generation_config = { "temperature": temperature, "top_p": 0.65, "top_k": 360, "max_output_tokens": 8196, } model = genai.GenerativeModel( model_name="gemini-2.0-flash", generation_config=generation_config, ) # Angle dictionaries have been moved to angles/angle_data.py # Incluir las instrucciones del sistema en el prompt principal system_prompt = f"""You are a world-class copywriter, with expertise in crafting hooks, headlines, and subject lines that immediately capture the reader's attention, prompting them to open the email or continue reading. FORMAT RULES: - Each headline must start with number and period - One headline per line - No explanations or categories - Add a line break between each headline - Avoid unnecessary : symbols - Each headline must be a complete and intriguing sentence IMPORTANT ANGLE INSTRUCTIONS: - The selected angle MUST be applied to EVERY headline - The angle modifies HOW the formula is expressed, not its structure - Think of the angle as a "tone overlay" on the formula - The formula provides the structure, the angle provides the style - Both must work together seamlessly FORMAT EXAMPLE: 1. Titular 1. 2. Titular 2. 3. Titular 3. 4. Titular 4. 5. Titular 5. IMPORTANT: - Each headline must be unique and memorable - Avoid clichés and generalities - Maintain an intriguing but credible tone - Adapt speaking language from the audience - Focus on transformative benefits - Follow the selected angle style while maintaining formula structure""" # Iniciar el prompt con las instrucciones del sistema headlines_instruction = f"{system_prompt}\n\n" # Añadir instrucciones de ángulo solo si no es "NINGUNO" if selected_angle != "NINGUNO": headlines_instruction += f""" ÁNGULO PRINCIPAL: {selected_angle} INSTRUCCIONES DE ÁNGULO ESPECÍFICAS: {angles[selected_angle]["instruction"]} IMPORTANTE: El ángulo {selected_angle} debe aplicarse como una "capa de estilo" sobre la estructura de la fórmula: 1. Mantén la estructura base de la fórmula intacta 2. Aplica el tono y estilo del ángulo {selected_angle} 3. Asegura que cada elemento de la fórmula refleje el ángulo 4. El ángulo afecta al "cómo" se dice, no al "qué" se dice EJEMPLOS EXITOSOS DEL ÁNGULO {selected_angle}: """ for example in angles[selected_angle]["examples"]: headlines_instruction += f"- {example}\n" headlines_instruction += ( f"\nTu tarea es crear {number_of_headlines} titulares irresistibles para {target_audience} " f"que capturen la atención instantáneamente y generen curiosidad sobre {product}. " ) if selected_angle != "NINGUNO": headlines_instruction += f"IMPORTANTE: Cada titular DEBE seguir el ángulo {selected_angle} de manera clara y consistente.\n\n" headlines_instruction += ( f"Evita menciones obvias de {product} y enfócate en despertar interés genuino" ) if selected_angle != "NINGUNO": headlines_instruction += f" usando el ángulo seleccionado" headlines_instruction += ".\n\n" headlines_instruction += ( f"IMPORTANTE: Estudia cuidadosamente estos ejemplos de la fórmula seleccionada. " f"Cada ejemplo representa el estilo y estructura a seguir" ) if selected_angle != "NINGUNO": headlines_instruction += f", adaptados al ángulo {selected_angle}" headlines_instruction += ":\n\n" # Agregar 5 ejemplos aleatorios de la fórmula random_examples = random.sample(selected_formula['examples'], min(5, len(selected_formula['examples']))) headlines_instruction += "EJEMPLOS DE LA FÓRMULA A SEGUIR:\n" for i, example in enumerate(random_examples, 1): headlines_instruction += f"{i}. {example}\n" headlines_instruction += "\nINSTRUCCIONES ESPECÍFICAS:\n" headlines_instruction += "1. Mantén la misma estructura y longitud que los ejemplos anteriores\n" headlines_instruction += "2. Usa el mismo tono y estilo de escritura\n" headlines_instruction += "3. Replica los patrones de construcción de frases\n" headlines_instruction += "4. Conserva el nivel de especificidad y detalle\n" headlines_instruction += f"5. Adapta el contenido para {target_audience} manteniendo la esencia de los ejemplos\n\n" headlines_instruction += f"FÓRMULA A SEGUIR:\n{selected_formula['description']}\n\n" # CORRECTO (con indentación): if selected_angle != "NINGUNO": headlines_instruction += f""" RECORDATORIO FINAL: 1. Sigue la estructura de la fórmula seleccionada 2. Aplica el ángulo como una "capa de estilo" 3. Mantén la coherencia entre fórmula y ángulo 4. Asegura que cada titular refleje ambos elementos GENERA AHORA: Crea {number_of_headlines} titulares que sigan fielmente el estilo y estructura de los ejemplos mostrados. """ else: headlines_instruction += f""" GENERA AHORA: Crea {number_of_headlines} titulares que sigan fielmente el estilo y estructura de los ejemplos mostrados. """ # Enviar el mensaje al modelo (sin condiciones de imagen) chat_session = model.start_chat( history=[ { "role": "user", "parts": [headlines_instruction], }, ] ) response = chat_session.send_message("Genera los titulares siguiendo exactamente el estilo de los ejemplos mostrados.") return response.text # Configurar la interfaz de usuario con Streamlit st.set_page_config(page_title="Enchanted Hooks", layout="wide") # Leer el contenido del archivo manual.md with open("manual.md", "r", encoding="utf-8") as file: manual_content = file.read() # Mostrar el contenido del manual en el sidebar st.sidebar.markdown(manual_content) # Load CSS from file with open("styles/main.css", "r") as f: css = f.read() # Apply the CSS st.markdown(f"", unsafe_allow_html=True) # Centrar el título y el subtítulo st.markdown("
{generated_headlines}