MohanadAfiffy commited on
Commit
2d6cc10
Β·
verified Β·
1 Parent(s): 5e0e47b

New Enhanced UI/UX

Browse files
Files changed (3) hide show
  1. app.py +64 -157
  2. clients.py +104 -253
  3. styles.html +138 -46
app.py CHANGED
@@ -1,163 +1,70 @@
1
  import streamlit as st
2
- from streamlit_extras.switch_page_button import switch_page
3
- def apollo_page_switch():
4
- switch_to_apollo = st.button("Visit the page for Apollo extract input")
5
- if switch_to_apollo:
6
- switch_page("pages_Apollo_Extract")
7
 
8
- def BH_page_switch():
9
- switch_to_BH = st.button("Visit the page dedicated for Business hubs use case")
10
- if switch_to_BH:
11
- switch_page("pages_Business_Hubs")
12
- st.markdown(
13
- """
14
- <style>
15
- .higher-title {
16
- font-size:35px !important;
17
- color: #4A90E2; /* Shade of blue */
18
- text-align: center;
19
- font-weight: bold;
20
- margin-top: -20px; /* Adjust this value to control the height */
21
- }
22
- </style>
23
- """,
24
- unsafe_allow_html=True,
25
- )
 
 
26
 
27
- st.markdown('<p class="higher-title">SalesIntel AI</p>', unsafe_allow_html=True)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
28
 
29
- st.markdown("""
30
- <h2 style="color:navy;">Business Hubs</h2>
31
- <h3 style="color:deepskyblue;">Generate Emails - NGO Oriented Function 🌍</h3>
32
- <h4 style="color:gray;">Generates personalized Emails for NGO campaigns, you have two options to use it</h4>
33
- <h4 style="color:cornflowerblue;">With Scraping:</h4>
34
- <h6 style="color:darkolivegreen;">Best for Mass Emails with minimal initial data.</h6>
35
- <h3 style="color:steelblue;">Required Columns:</h3>
36
- <ul>
37
- <li><strong>πŸ‘€ First Name</strong></li>
38
- <li><strong>🏒 Company Name</strong></li>
39
- <li><strong>🌐 Domain</strong></li>
40
- <li><strong>πŸ“Œ Title</strong></li>
41
- <li><strong>πŸ”— Person LinkedIn URL</strong></li>
42
- <li><strong>πŸ”— Company website</strong></li>
43
- <li><strong>πŸ‘€ Email</strong></li>
44
- </ul>
45
- <a href="https://docs.google.com/spreadsheets/d/1jkTrPqjO9bNBYlyluIGNsclJ7qz2vOrS2HOCD94ZD2Q/edit?usp=sharing" target="_blank">Check the template from here</a>
46
- <h4 style="color:cornflowerblue;">Opt-Out of Scraping:</h4>
47
- <h6 style="color:darkolivegreen;">Best for Low Volume Emails, Perfect for detailed information and targeted approach.</h6>
48
- <h4 style="color:steelblue;">Required Columns:</h4>
49
- <ul>
50
- <li><strong>πŸ‘€ First Name</strong></li>
51
- <li><strong>🏒 Company Name</strong></li>
52
- <li><strong>🌐 Domain</strong></li>
53
- <li><strong>πŸ“ User Description</strong></li>
54
- <li><strong>πŸ‘€ Email</strong></li>
55
- </ul>
56
- <a href="https://docs.google.com/spreadsheets/d/1TDpsbAZ62ukrUUmYp_5nWfcljWf5JYcZEevP_d4LzIc/edit?usp=sharing" target="_blank">Check the template from here</a>
57
- <h3 style="color:deepskyblue;">🏒 Generate Emails for Industries/Startups</h3>
58
- <h4 style="color:gray;">Supports personalized email generation for specific industries or startups, you have two options to use it</h4>
59
- <h4 style="color:cornflowerblue;">With Scraping:</h4>
60
- <h6 style="color:darkolivegreen;">For comprehensive company info with limited initial data.</h6>
61
- <h3 style="color:steelblue;">Required Columns:</h3>
62
- <ul>
63
- <li><strong>πŸ”— Website</strong></li>
64
- <li><strong>🏒 Company Name</strong></li>
65
- <li><strong>πŸ‘€ First Name</strong></li>
66
- <li><strong>πŸ‘€ Email</strong></li>
67
- </ul>
68
- <a href="https://docs.google.com/spreadsheets/d/1h2pQQKlTja_G3IJZvw0xNdiyZqkSToTD4zxFvCNDDeQ/edit?usp=sharing" target="_blank">Check the template from here</a>
69
- <h4 style="color:cornflowerblue;">Opt-Out of Scraping:</h4>
70
- <h6 style="color:darkolivegreen;">For complete company descriptions at hand.</h6>
71
- <h4 style="color:steelblue;">Required Columns:</h4>
72
- <ul>
73
- <li><strong>πŸ”— Website</strong></li>
74
- <li><strong>🏒 Company Name</strong></li>
75
- <li><strong>πŸ‘€ First Name</strong></li>
76
- <li><strong>πŸ“ Company Description</strong></li>
77
- <li><strong>πŸ‘€ Email</strong></li>
78
- </ul>
79
- <a href="https://docs.google.com/spreadsheets/d/1YFuE3ZC31QIGy0YLNK_2v89y0NbektheXXI1QDNA3Ng/edit?usp=sharing" target="_blank">Check the template from here</a>
80
- <br>
81
- """,unsafe_allow_html=True)
82
- st.markdown("""
83
- <br>
84
- <br>
85
- """, unsafe_allow_html=True)
86
- BH_page_switch()
87
 
 
 
 
88
 
89
- st.markdown("-----------------------------------------------------")
90
-
91
-
92
- st.markdown("""
93
- <h2 style='color: navy;'>Sales Core team - Apollo -</h2>
94
- <h3 style='color: darkslategray;'>Data Processing Functions</h3>
95
- <h4 style='color: darkcyan;'>Company-Specific Client Function</h4>
96
- <p style='color: slategray;'>Upload a CSV with company names and websites.</p>
97
- <ul>
98
- <li><strong style='color: darkolivegreen;'>With Scraping:</strong>
99
- <ul>
100
- <li>Required: <code>Website</code>, <code>Company Name for Emails</code></li>
101
- <li>The tool scrapes for extra company information.</li>
102
- </ul>
103
- </li>
104
- <li><strong style='color: darkolivegreen;'>Opt Out:</strong>
105
- <ul>
106
- <li>Required: <code>Website</code>, <code>Company Name for Emails</code>, <code>Company Description</code> (used as <code>scraped_content</code>)</li>
107
- <li>Provide your own company descriptions; no scraping.</li>
108
- </ul>
109
- </li>
110
- </ul>
111
- <h4 style='color: darkcyan;'>User-Specific Client Function</h4>
112
- <p style='color: slategray;'>Upload a CSV with personal and company information.</p>
113
- <ul>
114
- <li><strong style='color: darkolivegreen;'>With Scraping:</strong>
115
- <ul>
116
- <li>Required: <code>First Name</code>, <code>Company Name for Emails</code>, <code>Title</code>, <code>Website</code>, <code>Last Name</code>, <code>Person Linkedin Url</code>, <code>Email</code></li>
117
- <li>The tool scrapes LinkedIn and other sources for additional information.</li>
118
- </ul>
119
- </li>
120
- <li><strong style='color: darkolivegreen;'>Opt Out:</strong>
121
- <ul>
122
- <li>Required: <code>First Name</code>, <code>Company Name for Emails</code>, <code>Person Linkedin Url</code>, <code>User Description</code> (used as <code>Scrapped Profile</code>), <code>Email</code></li>
123
- <li>Use your data without scraping additional information.</li>
124
- </ul>
125
- </li>
126
- </ul>
127
- <h4 style='color: darkcyan;'>Both Features Function</h4>
128
- <p style='color: slategray;'>Combine user and company-specific data processing.</p>
129
- <ul>
130
- <li><strong style='color: darkolivegreen;'>With Scraping:</strong>
131
- <ul>
132
- <li>Required: <code>First Name</code>, <code>Company Name for Emails</code>, <code>Title</code>, <code>Last Name</code>, <code>Person Linkedin Url</code>, <code>Website</code>, <code>Email</code></li>
133
- <li>Scrapes for comprehensive data on both individuals and companies.</li>
134
- </ul>
135
- </li>
136
- <li><strong style='color: darkolivegreen;'>Opt Out:</strong>
137
- <ul>
138
- <li>Required: <code>First Name</code>, <code>Company Name for Emails</code>, <code>Person Linkedin Url</code>, <code>Company Description</code> (used as <code>scraped_content</code>), <code>User Description</code> (used as <code>Scrapped Profile</code>), <code>Email</code></li>
139
- <li>Rely on user-provided descriptions for both individuals and companies.</li>
140
- </ul>
141
- </li>
142
- </ul>
143
- <br>
144
- """, unsafe_allow_html=True)
145
-
146
-
147
- apollo_page_switch()
148
-
149
- logo_url = "https://i.imgur.com/WYnv26e.jpeg" # Replace this with your image's direct URL
150
- st.markdown(
151
- f"""
152
- <style>
153
- .logo {{
154
- position: fixed;
155
- bottom: 5px;
156
- right: 5px;
157
- width: 100px; # Adjust width as needed
158
- }}
159
- </style>
160
- <img src="{logo_url}" class="logo">
161
- """,
162
- unsafe_allow_html=True,
163
- )
 
1
  import streamlit as st
2
+ import os
3
+ from clients import CompanySpecificClient, UserSpecificClient,hooks,RengagmentEmail
 
 
 
4
 
5
+ st.set_page_config(page_title="SalesIntel",layout="wide")
6
+ st.html("styles.html")
7
+ endpoint = os.getenv('blog_lead_endpoint')
8
+ st.html('<h1 class="title"> SalesIntel </h1>')
9
+ st.html('<h4 class="hero-subtitle"> Your AI Sales Companions for Success</h4>')
10
+ email_options = [
11
12
+ "jamesel@omdena.com",
13
14
15
16
17
18
19
20
+ ]
21
+ email_address = st.selectbox("**Introduce yourself to us**", email_options)
22
+ def main():
23
+ # First container for Hook2Lead
24
+ cols = st.columns(2)
25
 
26
+ with cols[0]:
27
+ with st.container(border=True):
28
+ st.html('<h3><span>Re-engagement Campaigns</span></h3>')
29
+ cols_internal=st.columns(2)
30
+ with cols_internal[0]:
31
+ if st.button("Hook2lead", help="The tool will match it with the leads. You can bring your hooks either a blog, AI announcement, or trend"):
32
+ hook_dialog(email_address)
33
+ with cols_internal[1]:
34
+ if st.button("let the AI hooks", help="The AI will summarize the conversation and action points for you and write an email to explore new use cases. You can bring old conversations with previous leads"):
35
+ rengage_lead()
36
+ with cols[1]:
37
+ with st.container(border=True):
38
+ st.html('<h3><span>Cold Campaigns</span></h3>')
39
+ cols = st.columns(2)
40
+ with cols[0]:
41
+ if st.button("Tailored for companies", help="Generate cold emails based on company offerings and Omdena's services"):
42
+ cold_organization_dialog()
43
+ with cols[1]:
44
+ if st.button("Tailored for executives", help="Generate cold emails based on executive achievements"):
45
+ cold_executive_dialog()
46
 
47
+ @st.dialog("Bring your hook , and we will match it with the leads", width="large")
48
+ def hook_dialog(email_address):
49
+ hooks(email_address)
50
+ @st.dialog("personalized cold emails for company offerings", width="large")
51
+ def cold_organization_dialog():
52
+ CompanySpecificClient(email_address)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
53
 
54
+ @st.dialog("personalized cold emails for executive achievements", width="large")
55
+ def cold_executive_dialog():
56
+ UserSpecificClient(email_address)
57
 
58
+ @st.dialog("Tailored emails for re-engaging leads", width="large")
59
+ def rengage_lead():
60
+ RengagmentEmail(email_address)
61
+ if __name__ == "__main__":
62
+ logo_url = "https://i.imgur.com/WYnv26e.jpeg" # Replace this with your image's direct URL
63
+ st.markdown(
64
+ f"""
65
+ <img src="{logo_url}" class="logo">
66
+ """,
67
+ unsafe_allow_html=True,
68
+ )
69
+
70
+ main()
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
clients.py CHANGED
@@ -5,18 +5,20 @@ Created on Mon Jan 1 11:20:18 2024
5
  @author: mohanadafiffy
6
  """
7
  import os
 
 
 
 
8
  import streamlit as st
9
  import pandas as pd
10
  import requests
11
- import os
12
- host=os.getenv("backend")
13
-
14
- CompanyBackendService=host+'/receive_companies/'
15
- UserBackendService=host+'/receive_users/'
16
- BothFeaturesService=host+'/receive_data/'
17
- NGOEmailsService=host+'/receive_ngo_emails/'
18
- IndustryEmailService=host+'/receive_industry_email/'
19
-
20
 
21
  def add_https_to_urls(df, column_name):
22
  """
@@ -222,200 +224,85 @@ def UserSpecificClient(email_receiver):
222
  else:
223
  st.error("Data transmission failed. Please verify that your file contains the labels 'Company' and 'Person Linkedin Url'. Additionally, ensure that your file is valid and contains records and try again, if the problem persists please contact us at [email protected]")
224
 
225
- def bothFeaturesFunction(email_receiver):
226
- input_data=None
227
- submitted=None
228
- column_selections = {}
229
- uploaded_file = st.file_uploader("Kindly upload a CSV file that includes the names and websites of the companies", type=["csv"],key="BothFeaturesUploader")
230
- opt_out_scraping = st.checkbox("Opt out of scraping",key="BothOptOut")
231
- with st.form(key='User_Form'):
232
- if uploaded_file is not None:
233
- try:
234
- # Detect file type and read accordingly
235
- file_type = uploaded_file.name.split('.')[-1]
236
- if file_type == 'csv':
237
- try:
238
- df = pd.read_csv(uploaded_file)
239
- except:
240
- df = pd.read_csv(uploaded_file, encoding='ISO-8859-1')
241
- # Check if 'Person Linkedin Url' column exists
242
- required_essential_columns = ['First Name','Company Name for Emails','Email']
243
- missing_essential_columns = [col for col in required_essential_columns if col not in df.columns]
244
- required_scraping_columns=['Title','Last Name','Person Linkedin Url','Website']
245
- missing_scraping_columns = [col for col in required_scraping_columns if col not in df.columns]
246
- for col in missing_essential_columns:
247
- all_columns = df.columns.tolist()
248
- selected_column = st.selectbox(f"Select the column for {col}:", all_columns,key=col)
249
- column_selections[col] = selected_column
250
- # Generate selectboxes for missing scraping columns if not opting out
251
- if not opt_out_scraping:
252
- for col in missing_scraping_columns:
253
- all_columns = df.columns.tolist()
254
- selected_column = st.selectbox(f"Select the column for {col}:", all_columns, key=col)
255
- column_selections[col] = selected_column
256
- # Process the column renaming based on the selections
257
- for col, selected_column in column_selections.items():
258
- df.rename(columns={selected_column: col}, inplace=True)
259
-
260
- if opt_out_scraping:
261
- if 'Company Description' not in df.columns:
262
- all_columns = df.columns.tolist()
263
- description_column = st.selectbox("Select the column for Company Description:", all_columns,key="bothCompanyDescription")
264
- df.rename(columns={description_column: 'scraped_content'}, inplace=True)
265
- else:
266
- df.rename(columns={'Company Description': 'scraped_content'}, inplace=True)
267
- if 'User Description' not in df.columns:
268
- all_columns = df.columns.tolist()
269
- description_column = st.selectbox("Select the column for User Description:", all_columns,key="bothuserdescription")
270
- df.rename(columns={description_column: 'Scrapped Profile'}, inplace=True)
271
- else:
272
- df.rename(columns={'User Description': 'Scrapped Profile'}, inplace=True)
273
- # Check if "Person Linkedin Url" is in the DataFrame
274
- if 'Person Linkedin Url' not in df.columns:
275
- # Use the DataFrame index to generate a unique value for each row
276
- # You can adjust this to create a more complex identifier
277
- df['Person Linkedin Url'] = 'LI_' + df.index.astype(str)
278
- input_data = df
279
-
280
- except Exception as E:
281
- st.write(E)
282
- st.error("An error occurred while processing the file")
283
- # If the button is clicked, it will return True for this run
284
- prompt_notes= st.text_input("If applicable please mention the network name",key="CompanyPromptNotes")
285
- button_clicked = st.form_submit_button("Submit")
286
 
287
- # Update session state for the button
288
- if button_clicked:
289
- submitted = True
 
 
 
 
 
 
 
 
290
 
291
- # Use the session state variable to determine if the button was previously clicked
292
- if submitted and input_data is not None:
293
- df = input_data
294
- df = df.drop_duplicates(subset="Person Linkedin Url", keep='first')
295
- if opt_out_scraping:
296
- df=df[['First Name','Person Linkedin Url','Scrapped Profile',"Company Name for Emails","scraped_content","Email"]]
297
- else:
298
- df=df[['First Name', 'Last Name', 'Title', 'Person Linkedin Url',"Website","Company Name for Emails","Email"]]
299
- df=add_https_to_urls(df, 'Website')
300
-
301
- df = df.dropna().loc[~(df == '').all(axis=1)]
302
-
303
- st.write(df)
304
- # Convert DataFrame to CSV for transmission
305
- csv = df.to_csv(index=False)
306
 
307
- # Construct the data to send
308
- data_to_send = {"prompt_notes": prompt_notes, "dataframe": csv,"email_receiver":email_receiver,"filename": uploaded_file.name}
 
309
 
310
- # Sending the POST request to FastAPI
311
- response = requests.post(BothFeaturesService, json=data_to_send)
 
312
 
313
- if response.status_code == 200:
314
- st.info(f"We're processing your request. You can close the app now. An email will be sent to {email_receiver} once the process is finished.")
315
- else:
316
- st.error("Data transmission failed. Please verify that your file contains the labels 'Company' and 'Person Linkedin Url'. Additionally, ensure that your file is valid and contains records and try again, if the problem persists please contact us at [email protected]")
317
 
318
- def BH_Ngo(email_receiver,calendly_link,sender_name):
319
- input_data=None
320
- submitted=None
321
- column_selections = {}
322
- uploaded_file = st.file_uploader("Kindly upload a CSV file that includes the names and websites of the companies", type=["csv"],key="BothFeaturesUploader")
323
- opt_out_scraping = st.checkbox("Opt out of scraping",key="BothOptOut")
324
- with st.form(key='User_Form'):
325
- if uploaded_file is not None:
326
- try:
327
- # Detect file type and read accordingly
328
- file_type = uploaded_file.name.split('.')[-1]
329
- if file_type == 'csv':
330
- try:
331
- df = pd.read_csv(uploaded_file)
332
- except:
333
- df = pd.read_csv(uploaded_file, encoding='ISO-8859-1')
334
- # Check if 'Person Linkedin Url' column exists
335
- required_essential_columns = ['First Name','Company Name for Emails','Domain','Email']
336
- missing_essential_columns = [col for col in required_essential_columns if col not in df.columns]
337
- required_scraping_columns=['Title','Person Linkedin Url','Website']
338
- missing_scraping_columns = [col for col in required_scraping_columns if col not in df.columns]
339
- for col in missing_essential_columns:
340
- all_columns = df.columns.tolist()
341
- selected_column = st.selectbox(f"Select the column for {col}:", all_columns,key=col)
342
- column_selections[col] = selected_column
343
- # Generate selectboxes for missing scraping columns if not opting out
344
- if not opt_out_scraping:
345
- for col in missing_scraping_columns:
346
- all_columns = df.columns.tolist()
347
- selected_column = st.selectbox(f"Select the column for {col}:", all_columns, key=col)
348
- column_selections[col] = selected_column
349
- # Process the column renaming based on the selections
350
- for col, selected_column in column_selections.items():
351
- df.rename(columns={selected_column: col}, inplace=True)
352
-
353
- if opt_out_scraping:
354
- if 'User Description' not in df.columns:
355
- all_columns = df.columns.tolist()
356
- User_description_column = st.selectbox("Select the column for User Description:", all_columns,key="bothuserdescription")
357
- df.rename(columns={User_description_column: 'Scrapped Profile'}, inplace=True)
358
- else:
359
- df.rename(columns={'User Description': 'Scrapped Profile'}, inplace=True)
360
- # Check if "Person Linkedin Url" is in the DataFrame
361
- if 'Person Linkedin Url' not in df.columns:
362
- # Use the DataFrame index to generate a unique value for each row
363
- # You can adjust this to create a more complex identifier
364
- df['Person Linkedin Url'] = 'LI_' + df.index.astype(str)
365
- input_data = df
366
-
367
- except Exception as E:
368
- st.write(E)
369
- st.error("An error occurred while processing the file")
370
- # If the button is clicked, it will return True for this run
371
- button_clicked = st.form_submit_button("Submit")
372
 
373
- # Update session state for the button
374
- if button_clicked:
375
- submitted = True
376
 
377
- # Use the session state variable to determine if the button was previously clicked
378
- if submitted and input_data is not None:
379
- df = input_data
380
 
381
- df = df.drop_duplicates(subset="Person Linkedin Url", keep='first')
382
- if opt_out_scraping:
383
- df=df[['First Name','Person Linkedin Url','Scrapped Profile',"Company Name for Emails","Domain","Email"]]
384
-
385
- else:
386
-
387
- columns_to_select = ['First Name', 'Title', 'Person Linkedin Url', "Company Name for Emails", "Domain","Website","Email"]
388
- df=add_https_to_urls(df, 'Website')
389
- if 'Last Name' in df.columns:
390
- columns_to_select.insert(1, 'Last Name') # Insert 'Last Name' at the correct position
391
-
392
- df = df[columns_to_select]
393
-
394
-
395
- # Convert DataFrame to CSV for transmission
396
- df = df.dropna().loc[~(df == '').all(axis=1)]
397
- st.write(df)
398
- csv = df.to_csv(index=False)
399
 
400
- # Construct the data to send
401
- data_to_send = {"dataframe": csv, "email_receiver": email_receiver,"calendly_link":calendly_link,"sender_name":sender_name}
 
 
 
402
 
403
- # Sending the POST request to FastAPI
404
- response = requests.post(NGOEmailsService, json=data_to_send)
405
 
406
- if response.status_code == 200:
407
- st.info(f"We're processing your request. You can close the app now. An email will be sent to {email_receiver} once the process is finished.")
 
 
 
408
  else:
409
- st.error("Data transmission failed. Please verify that your file contains the labels 'Company' and 'Person Linkedin Url'. Additionally, ensure that your file is valid and contains records and try again, if the problem persists please contact us at [email protected]")
410
-
411
- def BH_industry(email_receiver,calendly_link,sender_name):
412
- input_data_companies = None
413
- submitted_companies = False
414
- uploaded_file = st.file_uploader("Kindly upload a CSV file that includes the names and websites of the companies", type=["csv"],key="CompanyUploader")
415
- opt_out_scraping = st.checkbox("Opt out of scraping",key="CompanyScraper")
416
- with st.form(key='Comapny_form'):
417
- if uploaded_file is not None:
418
 
 
 
419
  try:
420
  # Detect file type and read accordingly
421
  file_type = uploaded_file.name.split('.')[-1]
@@ -423,81 +310,45 @@ def BH_industry(email_receiver,calendly_link,sender_name):
423
  df = pd.read_csv(uploaded_file)
424
  elif file_type == 'xlsx':
425
  df = pd.read_excel(uploaded_file)
426
- # Check if 'Website' column exists
427
- if 'Website' not in df.columns:
428
- all_columns = df.columns.tolist()
429
- website_column = st.selectbox("Select the column for Website:", all_columns,key="CompanyWebsite")
430
- else:
431
- website_column = 'Website'
432
- if 'First Name' not in df.columns:
433
- all_columns = df.columns.tolist()
434
- name_column = st.selectbox("Select the column for first name:", all_columns,key="firstname")
435
- else:
436
- name_column = 'First Name'
437
- # Check if 'Company Name for Emails' column exists
438
- if 'Company Name for Emails' not in df.columns:
439
- all_columns = df.columns.tolist()
440
- company_column= st.selectbox("Select the column for Company Name :", all_columns,key="CompanyName")
441
- else:
442
- company_column = 'Company Name for Emails'
443
-
444
- if 'Email' not in df.columns:
445
- all_columns = df.columns.tolist()
446
- Email_column= st.selectbox("Select the column for email:", all_columns,key="Companyemail")
447
  else:
448
- Email_column = 'Email'
449
- if opt_out_scraping:
450
- if 'Company Description' not in df.columns:
451
- all_columns = df.columns.tolist()
452
- description_column = st.selectbox("Select the column for Description:", all_columns,key="CompanyDescription")
453
- df.rename(columns={description_column: 'scraped_content'}, inplace=True)
454
- else:
455
- df.rename(columns={'Company Description': 'scraped_content'}, inplace=True)
456
-
457
- input_data_companies = df
458
 
459
- except Exception as E :
460
- st.error("An error occured while processing the file")
461
-
462
- # Fetch the filtered data
463
-
464
 
465
  # If the button is clicked, it will return True for this run
466
  button_clicked = st.form_submit_button("Submit for processing")
467
 
468
- # 2. Update session state for the button
469
  if button_clicked:
470
- submitted_companies = True
471
- # 3. Use the session state variable to determine if the button was previously clicked
472
- if submitted_companies and input_data_companies is not None:
473
- df = input_data_companies
474
- if not opt_out_scraping:
475
- df[website_column] = df[website_column].astype(str)
476
- df=df[[website_column,company_column,name_column,Email_column]]
477
- df.columns = ["Website","Company Name for Emails","First Name","Email"]
478
- df = df.drop_duplicates(subset="Email", keep='first')
479
- df = df.dropna().loc[~(df == '').all(axis=1)]
480
- else:
481
- df[website_column] = df[website_column].astype(str)
482
- df=df[[website_column,company_column,"scraped_content",name_column,Email_column]]
483
- df.columns = ["Website","Company Name for Emails","scraped_content","First Name","Email"]
484
- df = df.drop_duplicates(subset="Email", keep='first')
485
- df = df.dropna().loc[~(df == '').all(axis=1)]
486
-
487
- df = df.dropna().loc[~(df == '').all(axis=1)]
488
- df=add_https_to_urls(df, 'Website')
489
- st.write(df)
490
  # Convert DataFrame to CSV for transmission
491
  csv = df.to_csv(index=False)
492
-
493
  # Construct the data to send
494
- data_to_send = {"dataframe": csv, "email_receiver": email_receiver,"calendly_link":calendly_link,"sender_name":sender_name}
495
-
496
  # Sending the POST request to FastAPI
497
- response = requests.post(IndustryEmailService, json=data_to_send)
498
-
499
  if response.status_code == 200:
500
  st.info(f"We're processing your request. You can close the app now. An email will be sent to {email_receiver} once the process is finished.")
501
  else:
502
- st.error("Data transmission failed. Please verify that your file contains the labels 'Company Website' and 'Company Name'. Additionally, ensure that your file is valid and contains records and try again , if the problem persists please contact us at mohanad@omdena.com")
503
- return None
 
 
5
  @author: mohanadafiffy
6
  """
7
  import os
8
+ from dotenv import load_dotenv
9
+
10
+ # Load environment variables from a .env file
11
+ load_dotenv()
12
  import streamlit as st
13
  import pandas as pd
14
  import requests
15
+ cold_host = os.getenv("backend_cold")
16
+ hook_host = os.getenv("hook_host") # Corrected here
17
+ rengagment_host = os.getenv("rengagement_host")
18
+ CompanyBackendService=cold_host+'/receive_companies/'
19
+ UserBackendService=cold_host+'/receive_users/'
20
+ RengagementBackendService=rengagment_host+'/query/'
21
+ HookBackendService=hook_host+'/query'
 
 
22
 
23
  def add_https_to_urls(df, column_name):
24
  """
 
224
  else:
225
  st.error("Data transmission failed. Please verify that your file contains the labels 'Company' and 'Person Linkedin Url'. Additionally, ensure that your file is valid and contains records and try again, if the problem persists please contact us at [email protected]")
226
 
227
+ def hooks(email_address):
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
228
 
229
+ account_stages = [
230
+ "Pre-Sales",
231
+ "Pre-Sales (Unresponsive, After Call)",
232
+ "Pre-Sales (Long-Term/ Cold)",
233
+ "Sales Opportunity",
234
+ "Closed Lost (Opportunity)",
235
+ "Current Client",
236
+ "Pre-Sales (Short-Term/ Hot)",
237
+ "Pre-Sales (Mid-Term/ Warm)",
238
+ "Project Cancelled"
239
+ ]
240
 
241
+ query_types = ["blog", "announcement", "AI_trend"]
 
 
 
 
 
 
 
 
 
 
 
 
 
 
242
 
243
+ # Get the current number of queries from query params
244
+ if 'num_queries' not in st.session_state:
245
+ st.session_state.num_queries = 1
246
 
247
+ # Creating a form
248
+ with st.form(key='blog2lead_form'):
249
+ selected_stages = st.multiselect("**Select Account Stages (optional)**", options=account_stages,key="Account stages multi-select")
250
 
251
+ queries = {}
252
+ query_types_selected = []
 
 
253
 
254
+ # Add query fields based on the current number of queries
255
+ for i in range(st.session_state.num_queries):
256
+ cols = st.columns([3, 1]) # Adjust the width ratio here
257
+ with cols[0]:
258
+ query_label = ["first", "second", "third", "fourth", "fifth", "sixth", "seventh", "eighth", "ninth", "tenth"][i]
259
+ query = st.text_input(f"**Enter your {query_label} hook**", key=f"query_{i}", help="you can enter your hook directly or a url")
260
+ with cols[1]:
261
+ query_type = st.selectbox(f"**Select {query_label} hook type**", query_types, key=f"type_{i}")
262
+ if query.strip():
263
+ queries[query] = query_type
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
264
 
265
+ # Button to add more query fields
266
+ add_query = st.form_submit_button(label='Add another query')
267
+ submit_button = st.form_submit_button(label='Submit')
268
 
269
+ if add_query:
270
+ st.session_state.num_queries += 1
 
271
 
272
+ if submit_button:
273
+ if queries and email_address:
274
+ # Define your data payload to send
275
+ queries = {k: v for k, v in queries.items() if k and v}
276
+ data_to_send = {
277
+ "queries": queries,
278
+ "email_receiver": email_address,
279
+ }
 
 
 
 
 
 
 
 
 
 
280
 
281
+ # Add the filter to the payload only if selected_stages is not empty
282
+ if selected_stages:
283
+ data_to_send["filter"] = {
284
+ "Account Stage": {"$in": selected_stages}
285
+ }
286
 
287
+ # Sending the POST request to FastAPI
288
+ response = requests.post(HookBackendService, json=data_to_send)
289
 
290
+ # Handling the response
291
+ if response.status_code == 200:
292
+ st.info("Your request has been processed successfully.")
293
+ else:
294
+ st.error("Data transmission failed. Please try again later.")
295
  else:
296
+ st.error("Please fill out all fields.")
297
+
298
+
299
+ def RengagmentEmail(email_receiver):
300
+ input_data_emails = None
301
+ submitted_emails = False
302
+ uploaded_file = st.file_uploader("Kindly upload a CSV file that includes the required columns", type=["csv"], key="Re-engagment email file uploader")
 
 
303
 
304
+ with st.form(key='Email_form'):
305
+ if uploaded_file is not None:
306
  try:
307
  # Detect file type and read accordingly
308
  file_type = uploaded_file.name.split('.')[-1]
 
310
  df = pd.read_csv(uploaded_file)
311
  elif file_type == 'xlsx':
312
  df = pd.read_excel(uploaded_file)
313
+
314
+ # Check if required columns exist
315
+ required_columns = ['To Email', 'Subject', 'Body HTML', 'Reply Message', 'To Company', 'website']
316
+ missing_columns = [col for col in required_columns if col not in df.columns]
317
+
318
+ if missing_columns:
319
+ st.error(f"Missing columns: {', '.join(missing_columns)}")
 
 
 
 
 
 
 
 
 
 
 
 
 
 
320
  else:
321
+ input_data_emails = df
 
 
 
 
 
 
 
 
 
322
 
323
+ except Exception as E:
324
+ st.error("An error occurred while processing the file")
 
 
 
325
 
326
  # If the button is clicked, it will return True for this run
327
  button_clicked = st.form_submit_button("Submit for processing")
328
 
329
+ # Update session state for the button
330
  if button_clicked:
331
+ submitted_emails = True
332
+
333
+ # Use the session state variable to determine if the button was previously clicked
334
+ if submitted_emails and input_data_emails is not None:
335
+ df = input_data_emails
336
+ df = df.dropna().loc[~(df == '').all(axis=1)]
337
+
338
+ st.write(df)
339
+
 
 
 
 
 
 
 
 
 
 
 
340
  # Convert DataFrame to CSV for transmission
341
  csv = df.to_csv(index=False)
342
+
343
  # Construct the data to send
344
+ data_to_send = {"dataframe": csv, "email_receiver": email_receiver, "filename": uploaded_file.name}
345
+
346
  # Sending the POST request to FastAPI
347
+ response = requests.post(RengagementBackendService, json=data_to_send)
348
+
349
  if response.status_code == 200:
350
  st.info(f"We're processing your request. You can close the app now. An email will be sent to {email_receiver} once the process is finished.")
351
  else:
352
+ st.error("Data transmission failed. Please verify that your file contains the required columns and try again. If the problem persists, please contact us.")
353
+
354
+ return None
styles.html CHANGED
@@ -1,68 +1,160 @@
1
  <style>
2
- /* General form styling */
3
- .stForm {
4
- background-color: #f9f9f9;
5
- padding: 20px;
6
- border-radius: 10px;
7
- box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
8
  }
9
-
10
- /* Label styling */
11
- .stForm label {
12
- font-weight: bold;
13
- color: #333;
14
  }
15
 
 
 
 
 
 
 
 
16
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
17
 
 
 
 
 
 
 
 
 
 
18
 
19
- /* Button styling */
20
- .stFormSubmitButton button {
21
- background-color: #4CAF50;
22
- color: white;
23
- padding: 10px 20px;
24
- border: none;
25
- border-radius: 5px;
26
- cursor: pointer;
27
- font-size: 16px;
 
 
 
 
 
28
  }
29
 
30
- .stFormSubmitButton button:hover {
31
- background-color: #45a049;
 
 
 
 
 
32
  }
33
 
34
- /* Column styling */
35
- .st-emotion-cache-ml2xh6, .st-emotion-cache-12w0qpk {
36
- padding: 0 10px;
 
 
 
 
37
  }
38
 
39
- /* Header styling */
40
- .stTitle {
41
- font-size: 2.5em;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
42
  font-weight: 700;
43
- color: #4CAF50;
44
- text-align: left;
45
- margin-bottom: 10px;
46
- text-decoration: underline;
47
- text-decoration-color: #4CAF50;
48
- transition: color 0.3s ease;
49
  }
50
 
51
- .stTitle:hover {
52
- color: #388E3C;
 
 
53
  }
54
 
55
- /* Subheader styling */
56
- .stSubheader {
57
- font-size: 1.5em;
58
- font-weight: 400;
59
- color: #555;
60
- text-align: left;
61
- margin-bottom: 20px;
62
- transition: color 0.3s ease;
63
  }
64
 
65
- .stSubheader:hover {
66
- color: #333;
67
  }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
68
  </style>
 
1
  <style>
2
+ @font-face {
3
+ font-family: PermanentMarker;
4
+ src: url(PermanentMarker-Regular.ttf);
 
 
 
5
  }
6
+
7
+ @font-face {
8
+ font-family: "Open Sans";
9
+ src: url(app/static/OpenSans-Regular.ttf);
 
10
  }
11
 
12
+ .title {
13
+ font-family: "Montserrat", sans-serif;
14
+ color: #000000;
15
+ font-size: 28px;
16
+ margin: 0px 0px 8px 0px;
17
+ padding: 0px;
18
+ }
19
 
20
+ h1 {
21
+ font-family: "Montserrat", sans-serif;
22
+ color: #000000;
23
+ font-size: 28px;
24
+ margin: 0px 0px 8px 0px;
25
+ padding: 30px 0px 0px 0px;
26
+ }
27
+ h3 {
28
+ font-family: "Open Sans", sans-serif;
29
+ font-size: 1.1em;
30
+ font-weight: 700;
31
+ color: #174C4F;
32
+ }
33
+ .hero-subtitle {
34
+ font-family: "Montserrat", sans-serif;
35
+ color: #FF6666;
36
+ font-size: 16px;
37
+ margin: 0px 0px 8px 50px; /* Adds 20px margin to the left */
38
+ padding-right: 100px;
39
+ font-style: italic; /* Adjust padding as needed */
40
+ }
41
+ div[data-testid="stVerticalBlockBorderWrapper"]:has(.stHtml > .watchlist_card) {
42
+ padding-bottom: 0em;
43
+ }
44
 
45
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .watchlist_symbol_name) {
46
+ & p {
47
+ color: #174C4F;
48
+ font-family: "Open Sans", sans-serif;
49
+ font-size: 1em;
50
+ font-weight: 700;
51
+ margin-bottom: 0;
52
+ }
53
+ }
54
 
55
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .watchlist_ticker) {
56
+ text-align: right;
57
+
58
+ & p {
59
+ font-size: 0.8em;
60
+ margin-bottom: 0;
61
+ }
62
+ }
63
+
64
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .watchlist_price_label) {
65
+ & p {
66
+ font-size: 0.8em;
67
+ margin-bottom: 0;
68
+ }
69
  }
70
 
71
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .watchlist_price_value) {
72
+ & p {
73
+ color: #174C4F;
74
+ font-family: "Open Sans", sans-serif;
75
+ font-size: 1.2em;
76
+ margin-bottom: 0;
77
+ }
78
  }
79
 
80
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .column_plotly) {
81
+ & .stPlotlyChart {
82
+ margin-top: 1em;
83
+ padding: 1em;
84
+ border-radius: 16px;
85
+ box-shadow: 0px 0px 10px rgba(81, 85, 195, 0.2);
86
+ }
87
  }
88
 
89
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .column_indicator) {
90
+ margin-top: 2.5em;
91
+ padding-left: 4em;
92
+ }
93
+
94
+ /* Adapted from https://startbootstrap.com/theme/sb-admin-2 */
95
+ div[data-testid="stMetric"] {
96
+ background-color: #FFFFFF;
97
+ border: 1px solid #CCCCCC;
98
+ padding: 1em 2em;
99
+ border-radius: 5px;
100
+ box-shadow: 0 0.15rem 1.75rem 0 rgba(58, 59, 69, 0.15);
101
+ }
102
+
103
+ label[data-testid="stMetricLabel"] p {
104
+ font-size: 1em;
105
+ }
106
+
107
+ div[data-testid="stMetricValue"] {
108
+ font-size: 1.3em;
109
  font-weight: 700;
110
+ color: #174C4F;
 
 
 
 
 
111
  }
112
 
113
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .low_indicator) {
114
+ & div[data-testid="stMetric"] {
115
+ border-left: 0.5rem solid #FF6666;
116
+ }
117
  }
118
 
119
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .high_indicator) {
120
+ & div[data-testid="stMetric"] {
121
+ border-left: 0.5rem solid rgb(15, 56, 109);
122
+ }
 
 
 
 
123
  }
124
 
125
+ div[data-testid="stVerticalBlock"]:has(> div > .stHtml > .bottom_indicator) {
126
+ margin-top: 0.5em;
127
  }
128
+
129
+ /* Custom button styling */
130
+ .stButton button, .stFormSubmitButton button {
131
+ background: linear-gradient(90deg, #3A8DFF 0%, #7B61FF 100%); /* Primary color */
132
+ color: white;
133
+ padding: 12px 24px;
134
+ border: none;
135
+ border-radius: 8px;
136
+ cursor: pointer;
137
+ font-size: 18px;
138
+ font-family: 'PermanentMarker', sans-serif; /* Change font */
139
+ font-weight: bold; /* Make font bold */
140
+ letter-spacing: 1px; /* Add letter spacing */
141
+ transition: background-color 0.3s ease, box-shadow 0.3s ease;
142
+ }
143
+
144
+ .stButton button:hover {
145
+ background-color: #093031; /* Darker color on hover */
146
+ box-shadow: 0 6px 12px rgba(0, 0, 0, 0.2); /* Add shadow on hover */
147
+ }
148
+
149
+ .stButton button:active {
150
+ background: linear-gradient(90deg, #3A8DFF 0%, #7B61FF 100%); /* Primary color */
151
+ box-shadow: 0 3px 6px rgba(0, 0, 0, 0.2); /* Reduce shadow on click */
152
+ }
153
+ .logo {
154
+ position: fixed;
155
+ bottom: 10px;
156
+ right: 5px;
157
+ width: 100px; /* Adjust width as needed */
158
+ }
159
+
160
  </style>