Name: Towards AI Legal Name: Towards AI, Inc. Description: Towards AI is the world's leading artificial intelligence (AI) and technology publication. Read by thought-leaders and decision-makers around the world. Phone Number: +1-650-246-9381 Email: pub@towardsai.net
228 Park Avenue South New York, NY 10003 United States
Website: Publisher: https://towardsai.net/#publisher Diversity Policy: https://towardsai.net/about Ethics Policy: https://towardsai.net/about Masthead: https://towardsai.net/about
Name: Towards AI Legal Name: Towards AI, Inc. Description: Towards AI is the world's leading artificial intelligence (AI) and technology publication. Founders: Roberto Iriondo, , Job Title: Co-founder and Advisor Works for: Towards AI, Inc. Follow Roberto: X, LinkedIn, GitHub, Google Scholar, Towards AI Profile, Medium, ML@CMU, FreeCodeCamp, Crunchbase, Bloomberg, Roberto Iriondo, Generative AI Lab, Generative AI Lab Denis Piffaretti, Job Title: Co-founder Works for: Towards AI, Inc. Louie Peters, Job Title: Co-founder Works for: Towards AI, Inc. Louis-François Bouchard, Job Title: Co-founder Works for: Towards AI, Inc. Cover:
Towards AI Cover
Logo:
Towards AI Logo
Areas Served: Worldwide Alternate Name: Towards AI, Inc. Alternate Name: Towards AI Co. Alternate Name: towards ai Alternate Name: towardsai Alternate Name: towards.ai Alternate Name: tai Alternate Name: toward ai Alternate Name: toward.ai Alternate Name: Towards AI, Inc. Alternate Name: towardsai.net Alternate Name: pub.towardsai.net
5 stars – based on 497 reviews

Frequently Used, Contextual References

TODO: Remember to copy unique IDs whenever it needs used. i.e., URL: 304b2e42315e

Resources

Take our 85+ lesson From Beginner to Advanced LLM Developer Certification: From choosing a project to deploying a working product this is the most comprehensive and practical LLM course out there!

Publication

Optimizing Emergency Response in Calgary
Data Science   Data Visualization   Latest   Machine Learning

Optimizing Emergency Response in Calgary

Last Updated on April 7, 2024 by Editorial Team

Author(s): Vishnu Regimon Nair

Originally published on Towards AI.

Optimizing Emergency Response in Calgary

Photo by Constante Lim on Unsplash

Facing the Growing Challenge: Optimizing Emergency Response Times in Calgary

With the ever-increasing population and evolving cities, the demand and need for emergency services grow stronger. This project simulates the dynamics of emergency response by mapping the paths navigated and times taken by emergency services, including fire trucks and ambulances, to reach their required destination. This project aims to get help to you faster, which could mean fewer injuries, less property damage, and even saved lives. By understanding how quickly firefighters and ambulances can reach different parts of the city, we can improve Calgary’s emergency response system. This means getting the help you need quicker when it matters most.

Link to the Streamlit dashboard: https://calgary-emergency-response-times.streamlit.app/

Unlocking Efficiency: Mapping Response Times in Calgary’s Emergency Services

This project tackles that challenge by simulating emergency response times across Calgary. It calculates how long the closest emergency vehicle takes to reach various city areas, either from a fire station or an EMS station.

Response time is between your 911 call and the first fire truck or ambulance arriving. The faster the first responder gets there, especially in serious emergencies, the better the chances of a positive outcome.

Calgary can improve its emergency response system by understanding where response times are slower. We will identify neighborhoods with slower response times and recommend extra attention from emergency response planners. This could translate to faster help during critical moments, potentially saving lives.

Leveraging Open Data: The Foundation of the Analysis

To see how fast firefighters and ambulances can reach you, we looked at various sources of relevant information:

  1. EMS station locations: Locations of all EMS stations spread around the city.
  2. Fire station locations: This helps us understand how well fire stations are spread throughout Calgary.
  3. Addresses in Calgary: We used every address to ensure every neighborhood was included.
  4. Calgary roads: The road network in Calgary allows us to understand the paths vehicles take in the city.
  5. Communities of Calgary: Understanding Calgary’s diverse communities helps us paint a complete picture.
  6. City Boundary: This captures Calgary’s jurisdiction, showing the scope of our project.

Contains information licensed under the Open Government Licence — City of Calgary. License: https://data.calgary.ca/stories/s/u45n-7awa

Bridging the Gap: Acknowledging Real-World Factors in Response Time Modeling

This simulation helps us understand how fast emergency services can get to you in an emergency. A few assumptions are taken to simplify the analysis and also because of a need for more data resources. Certain factors like Traffic during the day or detours due to construction or weather can’t be proactively determined. By considering these limitations, we better understand what response times might be like. If many areas require assistance, even under these assumptions, it would take much longer to reach under real-world conditions. This is a great starting point, and the more details we factor in, the better we can understand how to get help to you faster in an emergency!

Traffic Jams: Even with flashing lights and sirens, ambulances can get stuck in traffic like everyone else. This can significantly delay their arrival time, especially during peak hours.

Busy Ambulances: Sometimes ambulances are already responding to other emergencies and might only be available after a while. Additionally, if a station has all its ambulances out on calls, they might need to call for help from another station, which can take extra time.

Not All Emergencies Are the Same: Some situations require a faster response than others. Dispatchers prioritize calls based on the severity of the emergency and the potential for life loss. This means a less critical situation has to wait longer if more severe calls are happening simultaneously.

Road Closures and Detours: Unexpected events like accidents, construction, or bad weather can force road closures and detours. This throws a wrench in the fastest route for ambulances and can significantly slow them down.

Rush Hour and Special Events: Traffic congestion isn’t constant. During rush hour or significant events like the Stampede, the roads get much busier, potentially causing delays for ambulances trying to reach you quickly.

Building the Framework: How We Calculated Response Times

We used a mapping tool called QGIS to analyze the city’s road network and emergency stations.

Here’s what we did:

  • Collected data: We gathered information on all the roads in Calgary and where fire stations and ambulances are located.
  • Mapped the best routes: We used QGIS and iso-area interpolation techniques to find the quickest paths ambulances could take, avoiding unrealistic shortcuts.
  • Focused on distance: While traffic and other factors can affect speed, we looked at how far ambulances must travel to reach different parts of the city.
  • Covered all of Calgary: This analysis included the entire city, so every neighborhood was included.

Calgary Fire Station Response Times

  1. Average Response Time by Forward Sortation Area (FSA)
  2. Response Time Distribution by Forward Sortation Area (FSA)
  3. Average Response Time by Community
  4. Response Time Distribution by Community

1) Average Response Time by Forward Sortation Area (FSA)

Image by Author
# Create a map
m = folium.Map(location=[51.0447,-114.0719], zoom_start=10)

for index, row in df_fire.iterrows():
folium.CircleMarker(
location=[row["LAT"], row["LON"]],
popup=row["NAME"],
tooltip=row["NAME"],
radius=2,
fill=True,
fill_color="red",
fill_opacity=1,
icon=folium.Icon(color="red"),
color="red",
opacity=1,
weight=1
).add_to(m)

# Add the shapefile with color ranges to the map
folium.GeoJson(
df_avgtimes_fire,
style_function=lambda x: {
"fillColor": color_scale(x["properties"]["Avg_time"]),
"color": "black",
"weight": 2,
"fillOpacity": 0.6,
},
tooltip=folium.GeoJsonTooltip(fields=["FSA", "Avg_time"], aliases=["FSA", "Avg Response Time(min)"], sticky=False),
highlight_function=lambda x: {
"weight": 4,
"fillOpacity": 0.9,
},
name="Average Response Time (s)"
).add_to(m)

The map pinpoints Forward Sortation Areas (FSAs) with the slowest average response times: T3S, T1X, and T3P. Residents in these areas might benefit from additional support to ensure timely emergency response.

The following bar chart shows the average Forward Sortation Area (FSA) response times in minutes.

Image by Author
fig_bar = px.bar(df_avgtimes_fire, x='FSA', y='Avg_time', labels={'FSA':'Forward Sortation Area', 'Avg_time':'Average Response Time (mins)'})
fig_bar.update_traces(marker_color='rgb(158,202,225)', marker_line_color='rgb(8,48,107)', marker_line_width=1.5, opacity=0.6)
fig_bar.update_layout(title_text='Average Emergency Response Times by FSA')
fig_bar.add_hline(y=7, line_dash="dash", line_color="red",
annotation_text="Target",annotation_font_color="red")
st.plotly_chart(fig_bar, use_container_width=True)

2) Response Time Distribution by Forward Sortation Area (FSA)

The bar chart reveals that 91% of FSAs consistently meet the 6-minute response target, with impressive percentages achieving even faster response times (2–3 minutes). This highlights the efficiency of fire services across most areas.

The following histogram shows the distribution of response times in minutes.

Image by Author
fig_hist = px.histogram(df_avgtimes_fire, x='Avg_time', nbins=20, labels={'Avg_time':'Average Response Time (mins)'})
fig_hist.update_traces(marker_color='rgb(158,202,225)', marker_line_color='rgb(8,48,107)', marker_line_width=1.5, opacity=0.6)
fig_hist.update_layout(title_text='Distribution of Emergency Response Times',yaxis_title = 'Count')
fig_hist.add_vline(x=7, line_dash="dash", line_color="red",
annotation_text="Target",annotation_font_color="red")
st.plotly_chart(fig_hist, use_container_width=True)

3) Average Response Time by Community

Image by Author

The map identifies communities with the slowest average response times: Glacier Ridge, Keystone Hills, Twin Hills, and Pegasus. These communities, excluding sub-areas, require closer attention to enhance emergency response coverage.

The following bar chart shows the average response times by Community in minutes.

Image by Author

4) Response Time Distribution by Community

The bar chart showcases that 93% of communities meet the 7-minute response target, with many achieving even faster response times (2–5 minutes). This demonstrates an overall swift response across most community areas.

The following histogram shows the distribution of response times in minutes.

Image by Author

Emergency Medical Services Response Times

  1. Average Response Time by Forward Sortation Area (FSA)
  2. Response Time Distribution by Forward Sortation Area (FSA)
  3. Average Response Time by Community
  4. Response Time Distribution by Community

1) Average Response Time by Forward Sortation Area (FSA)

T1X, T3S, T2X, and T2Y emerge as FSAs with the slowest average response times (Image by Author), potentially requiring focused initiatives to improve ambulance accessibility.

The following bar chart shows the average Forward Sortation Area (FSA) response times in minutes.

Image by Author

2) Response Time Distribution by Forward Sortation Area (FSA)

The histogram shows that 88% of FSAs meet the 8-minute and 59-second target, with many achieving faster response times (3–4 minutes). This indicates an overall emphasis on swift medical emergency response.

The following histogram shows the distribution of response times in minutes.

Image by Author

3) Average Response Time by Community

Image by Author

Glacier Ridge, Legacy, Hotchkiss, and Wolf Willow require attention as communities with the slowest average response times (excluding sub-areas, as depicted in the map).

The following bar chart shows the average response times by Community in minutes.

Image by Author

4) Response Time Distribution by Community

While the histogram reveals that over 14% of communities fall short of the 8-minute and 59-second target, continued efforts are needed to ensure timely medical emergency response in all areas.

The following histogram shows the distribution of response times in minutes.

Image by Author

Optimizing Calgary’s Response: Targeted Recommendations

The Calgary Fire Department (CFD) achieves its impressive goal of reaching 90% of emergencies within 7 minutes. The CFD should continue to commit to a swift response.

However, further action is needed on the EMS side to ensure continued excellence in fire and medical emergencies. Establishing additional EMS stations is crucial to keep pace with Calgary’s growth.

One potentially cost-effective and strategic solution involves co-locating EMS stations within existing fire stations. While feasibility needs careful consideration, exploring this option in stations 35(11280 Valley Ridge Blvd NW), 40(11920 Symons Valley Rd NW), and 43(969 Walden Dr. SE) could yield significant benefits.

Image by Author

Ensuring Equity in Response Times: Balancing Optimization with Ethics

Everyone deserves to get help fast in an emergency, no matter where they live or who they are. But sometimes, people in specific neighborhoods — especially those with more diversity — might wait longer for ambulances. We must avoid approaches that unintentionally disadvantage particular communities, particularly those with racial or ethnic diversity.

That’s not fair, and here’s what we should be doing to fix it:

  • Looking at the data: Use maps and numbers to determine where response times are slowest.
  • Talking to everyone: Work with different communities to understand their concerns.
  • Sending help where needed most: Place resources in areas with the longest waits.

Most importantly, everyone gets the help they need as quickly as possible. We must uphold the moral imperative that saving every life supersedes optimization goals. We have to ensure the emergency response system is fair and protects everyone.

Saving Lives Starts Now: A Call to Action for Improved Response Times

Imagine getting help faster in an emergency! That’s precisely what this project is working on, using maps and data analysis. This lets them see where ambulances take the longest to arrive so they can improve response times and save more lives. It’s about real people in our Community getting the help they need when they need it most. With this approach, Calgary can build a more robust emergency response system that keeps everyone safe.

Join thousands of data leaders on the AI newsletter. Join over 80,000 subscribers and keep up to date with the latest developments in AI. From research to projects and ideas. If you are building an AI startup, an AI-related product, or a service, we invite you to consider becoming a sponsor.

Published via Towards AI

JOIN NOW!

Gain exclusive access to top AI tutorials, courses, and books to elevate your skills.

    We won't send you spam. Unsubscribe at any time.

    Feedback ↓

    Sign Up for the Course
    `; } else { console.error('Element with id="subscribe" not found within the page with class "home".'); } } }); // Remove duplicate text from articles /* Backup: 09/11/24 function removeDuplicateText() { const elements = document.querySelectorAll('h1, h2, h3, h4, h5, strong'); // Select the desired elements const seenTexts = new Set(); // A set to keep track of seen texts const tagCounters = {}; // Object to track instances of each tag elements.forEach(el => { const tagName = el.tagName.toLowerCase(); // Get the tag name (e.g., 'h1', 'h2', etc.) // Initialize a counter for each tag if not already done if (!tagCounters[tagName]) { tagCounters[tagName] = 0; } // Only process the first 10 elements of each tag type if (tagCounters[tagName] >= 2) { return; // Skip if the number of elements exceeds 10 } const text = el.textContent.trim(); // Get the text content const words = text.split(/\s+/); // Split the text into words if (words.length >= 4) { // Ensure at least 4 words const significantPart = words.slice(0, 5).join(' '); // Get first 5 words for matching // Check if the text (not the tag) has been seen before if (seenTexts.has(significantPart)) { // console.log('Duplicate found, removing:', el); // Log duplicate el.remove(); // Remove duplicate element } else { seenTexts.add(significantPart); // Add the text to the set } } tagCounters[tagName]++; // Increment the counter for this tag }); } removeDuplicateText(); */ // Remove duplicate text from articles function removeDuplicateText() { const elements = document.querySelectorAll('h1, h2, h3, h4, h5, strong'); // Select the desired elements const seenTexts = new Set(); // A set to keep track of seen texts const tagCounters = {}; // Object to track instances of each tag // List of classes to be excluded const excludedClasses = ['medium-author', 'post-widget-title']; elements.forEach(el => { // Skip elements with any of the excluded classes if (excludedClasses.some(cls => el.classList.contains(cls))) { return; // Skip this element if it has any of the excluded classes } const tagName = el.tagName.toLowerCase(); // Get the tag name (e.g., 'h1', 'h2', etc.) // Initialize a counter for each tag if not already done if (!tagCounters[tagName]) { tagCounters[tagName] = 0; } // Only process the first 10 elements of each tag type if (tagCounters[tagName] >= 10) { return; // Skip if the number of elements exceeds 10 } const text = el.textContent.trim(); // Get the text content const words = text.split(/\s+/); // Split the text into words if (words.length >= 4) { // Ensure at least 4 words const significantPart = words.slice(0, 5).join(' '); // Get first 5 words for matching // Check if the text (not the tag) has been seen before if (seenTexts.has(significantPart)) { // console.log('Duplicate found, removing:', el); // Log duplicate el.remove(); // Remove duplicate element } else { seenTexts.add(significantPart); // Add the text to the set } } tagCounters[tagName]++; // Increment the counter for this tag }); } removeDuplicateText(); //Remove unnecessary text in blog excerpts document.querySelectorAll('.blog p').forEach(function(paragraph) { // Replace the unwanted text pattern for each paragraph paragraph.innerHTML = paragraph.innerHTML .replace(/Author\(s\): [\w\s]+ Originally published on Towards AI\.?/g, '') // Removes 'Author(s): XYZ Originally published on Towards AI' .replace(/This member-only story is on us\. Upgrade to access all of Medium\./g, ''); // Removes 'This member-only story...' }); //Load ionic icons and cache them if ('localStorage' in window && window['localStorage'] !== null) { const cssLink = 'https://code.ionicframework.com/ionicons/2.0.1/css/ionicons.min.css'; const storedCss = localStorage.getItem('ionicons'); if (storedCss) { loadCSS(storedCss); } else { fetch(cssLink).then(response => response.text()).then(css => { localStorage.setItem('ionicons', css); loadCSS(css); }); } } function loadCSS(css) { const style = document.createElement('style'); style.innerHTML = css; document.head.appendChild(style); } //Remove elements from imported content automatically function removeStrongFromHeadings() { const elements = document.querySelectorAll('h1, h2, h3, h4, h5, h6, span'); elements.forEach(el => { const strongTags = el.querySelectorAll('strong'); strongTags.forEach(strongTag => { while (strongTag.firstChild) { strongTag.parentNode.insertBefore(strongTag.firstChild, strongTag); } strongTag.remove(); }); }); } removeStrongFromHeadings(); "use strict"; window.onload = () => { /* //This is an object for each category of subjects and in that there are kewords and link to the keywods let keywordsAndLinks = { //you can add more categories and define their keywords and add a link ds: { keywords: [ //you can add more keywords here they are detected and replaced with achor tag automatically 'data science', 'Data science', 'Data Science', 'data Science', 'DATA SCIENCE', ], //we will replace the linktext with the keyword later on in the code //you can easily change links for each category here //(include class="ml-link" and linktext) link: 'linktext', }, ml: { keywords: [ //Add more keywords 'machine learning', 'Machine learning', 'Machine Learning', 'machine Learning', 'MACHINE LEARNING', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, ai: { keywords: [ 'artificial intelligence', 'Artificial intelligence', 'Artificial Intelligence', 'artificial Intelligence', 'ARTIFICIAL INTELLIGENCE', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, nl: { keywords: [ 'NLP', 'nlp', 'natural language processing', 'Natural Language Processing', 'NATURAL LANGUAGE PROCESSING', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, des: { keywords: [ 'data engineering services', 'Data Engineering Services', 'DATA ENGINEERING SERVICES', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, td: { keywords: [ 'training data', 'Training Data', 'training Data', 'TRAINING DATA', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, ias: { keywords: [ 'image annotation services', 'Image annotation services', 'image Annotation services', 'image annotation Services', 'Image Annotation Services', 'IMAGE ANNOTATION SERVICES', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, l: { keywords: [ 'labeling', 'labelling', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, pbp: { keywords: [ 'previous blog posts', 'previous blog post', 'latest', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, mlc: { keywords: [ 'machine learning course', 'machine learning class', ], //Change your article link (include class="ml-link" and linktext) link: 'linktext', }, }; //Articles to skip let articleIdsToSkip = ['post-2651', 'post-3414', 'post-3540']; //keyword with its related achortag is recieved here along with article id function searchAndReplace(keyword, anchorTag, articleId) { //selects the h3 h4 and p tags that are inside of the article let content = document.querySelector(`#${articleId} .entry-content`); //replaces the "linktext" in achor tag with the keyword that will be searched and replaced let newLink = anchorTag.replace('linktext', keyword); //regular expression to search keyword var re = new RegExp('(' + keyword + ')', 'g'); //this replaces the keywords in h3 h4 and p tags content with achor tag content.innerHTML = content.innerHTML.replace(re, newLink); } function articleFilter(keyword, anchorTag) { //gets all the articles var articles = document.querySelectorAll('article'); //if its zero or less then there are no articles if (articles.length > 0) { for (let x = 0; x < articles.length; x++) { //articles to skip is an array in which there are ids of articles which should not get effected //if the current article's id is also in that array then do not call search and replace with its data if (!articleIdsToSkip.includes(articles[x].id)) { //search and replace is called on articles which should get effected searchAndReplace(keyword, anchorTag, articles[x].id, key); } else { console.log( `Cannot replace the keywords in article with id ${articles[x].id}` ); } } } else { console.log('No articles found.'); } } let key; //not part of script, added for (key in keywordsAndLinks) { //key is the object in keywords and links object i.e ds, ml, ai for (let i = 0; i < keywordsAndLinks[key].keywords.length; i++) { //keywordsAndLinks[key].keywords is the array of keywords for key (ds, ml, ai) //keywordsAndLinks[key].keywords[i] is the keyword and keywordsAndLinks[key].link is the link //keyword and link is sent to searchreplace where it is then replaced using regular expression and replace function articleFilter( keywordsAndLinks[key].keywords[i], keywordsAndLinks[key].link ); } } function cleanLinks() { // (making smal functions is for DRY) this function gets the links and only keeps the first 2 and from the rest removes the anchor tag and replaces it with its text function removeLinks(links) { if (links.length > 1) { for (let i = 2; i < links.length; i++) { links[i].outerHTML = links[i].textContent; } } } //arrays which will contain all the achor tags found with the class (ds-link, ml-link, ailink) in each article inserted using search and replace let dslinks; let mllinks; let ailinks; let nllinks; let deslinks; let tdlinks; let iaslinks; let llinks; let pbplinks; let mlclinks; const content = document.querySelectorAll('article'); //all articles content.forEach((c) => { //to skip the articles with specific ids if (!articleIdsToSkip.includes(c.id)) { //getting all the anchor tags in each article one by one dslinks = document.querySelectorAll(`#${c.id} .entry-content a.ds-link`); mllinks = document.querySelectorAll(`#${c.id} .entry-content a.ml-link`); ailinks = document.querySelectorAll(`#${c.id} .entry-content a.ai-link`); nllinks = document.querySelectorAll(`#${c.id} .entry-content a.ntrl-link`); deslinks = document.querySelectorAll(`#${c.id} .entry-content a.des-link`); tdlinks = document.querySelectorAll(`#${c.id} .entry-content a.td-link`); iaslinks = document.querySelectorAll(`#${c.id} .entry-content a.ias-link`); mlclinks = document.querySelectorAll(`#${c.id} .entry-content a.mlc-link`); llinks = document.querySelectorAll(`#${c.id} .entry-content a.l-link`); pbplinks = document.querySelectorAll(`#${c.id} .entry-content a.pbp-link`); //sending the anchor tags list of each article one by one to remove extra anchor tags removeLinks(dslinks); removeLinks(mllinks); removeLinks(ailinks); removeLinks(nllinks); removeLinks(deslinks); removeLinks(tdlinks); removeLinks(iaslinks); removeLinks(mlclinks); removeLinks(llinks); removeLinks(pbplinks); } }); } //To remove extra achor tags of each category (ds, ml, ai) and only have 2 of each category per article cleanLinks(); */ //Recommended Articles var ctaLinks = [ /* ' ' + '

    Subscribe to our AI newsletter!

    ' + */ '

    Take our 85+ lesson From Beginner to Advanced LLM Developer Certification: From choosing a project to deploying a working product this is the most comprehensive and practical LLM course out there!

    '+ '

    Towards AI has published Building LLMs for Production—our 470+ page guide to mastering LLMs with practical projects and expert insights!

    ' + '
    ' + '' + '' + '

    Note: Content contains the views of the contributing authors and not Towards AI.
    Disclosure: This website may contain sponsored content and affiliate links.

    ' + 'Discover Your Dream AI Career at Towards AI Jobs' + '

    Towards AI has built a jobs board tailored specifically to Machine Learning and Data Science Jobs and Skills. Our software searches for live AI jobs each hour, labels and categorises them and makes them easily searchable. Explore over 10,000 live jobs today with Towards AI Jobs!

    ' + '
    ' + '

    🔥 Recommended Articles 🔥

    ' + 'Why Become an LLM Developer? Launching Towards AI’s New One-Stop Conversion Course'+ 'Testing Launchpad.sh: A Container-based GPU Cloud for Inference and Fine-tuning'+ 'The Top 13 AI-Powered CRM Platforms
    ' + 'Top 11 AI Call Center Software for 2024
    ' + 'Learn Prompting 101—Prompt Engineering Course
    ' + 'Explore Leading Cloud Providers for GPU-Powered LLM Training
    ' + 'Best AI Communities for Artificial Intelligence Enthusiasts
    ' + 'Best Workstations for Deep Learning
    ' + 'Best Laptops for Deep Learning
    ' + 'Best Machine Learning Books
    ' + 'Machine Learning Algorithms
    ' + 'Neural Networks Tutorial
    ' + 'Best Public Datasets for Machine Learning
    ' + 'Neural Network Types
    ' + 'NLP Tutorial
    ' + 'Best Data Science Books
    ' + 'Monte Carlo Simulation Tutorial
    ' + 'Recommender System Tutorial
    ' + 'Linear Algebra for Deep Learning Tutorial
    ' + 'Google Colab Introduction
    ' + 'Decision Trees in Machine Learning
    ' + 'Principal Component Analysis (PCA) Tutorial
    ' + 'Linear Regression from Zero to Hero
    '+ '

    ', /* + '

    Join thousands of data leaders on the AI newsletter. It’s free, we don’t spam, and we never share your email address. Keep up to date with the latest work in AI. From research to projects and ideas. If you are building an AI startup, an AI-related product, or a service, we invite you to consider becoming a sponsor.

    ',*/ ]; var replaceText = { '': '', '': '', '
    ': '
    ' + ctaLinks + '
    ', }; Object.keys(replaceText).forEach((txtorig) => { //txtorig is the key in replacetext object const txtnew = replaceText[txtorig]; //txtnew is the value of the key in replacetext object let entryFooter = document.querySelector('article .entry-footer'); if (document.querySelectorAll('.single-post').length > 0) { //console.log('Article found.'); const text = entryFooter.innerHTML; entryFooter.innerHTML = text.replace(txtorig, txtnew); } else { // console.log('Article not found.'); //removing comment 09/04/24 } }); var css = document.createElement('style'); css.type = 'text/css'; css.innerHTML = '.post-tags { display:none !important } .article-cta a { font-size: 18px; }'; document.body.appendChild(css); //Extra //This function adds some accessibility needs to the site. function addAlly() { // In this function JQuery is replaced with vanilla javascript functions const imgCont = document.querySelector('.uw-imgcont'); imgCont.setAttribute('aria-label', 'AI news, latest developments'); imgCont.title = 'AI news, latest developments'; imgCont.rel = 'noopener'; document.querySelector('.page-mobile-menu-logo a').title = 'Towards AI Home'; document.querySelector('a.social-link').rel = 'noopener'; document.querySelector('a.uw-text').rel = 'noopener'; document.querySelector('a.uw-w-branding').rel = 'noopener'; document.querySelector('.blog h2.heading').innerHTML = 'Publication'; const popupSearch = document.querySelector$('a.btn-open-popup-search'); popupSearch.setAttribute('role', 'button'); popupSearch.title = 'Search'; const searchClose = document.querySelector('a.popup-search-close'); searchClose.setAttribute('role', 'button'); searchClose.title = 'Close search page'; // document // .querySelector('a.btn-open-popup-search') // .setAttribute( // 'href', // 'https://medium.com/towards-artificial-intelligence/search' // ); } // Add external attributes to 302 sticky and editorial links function extLink() { // Sticky 302 links, this fuction opens the link we send to Medium on a new tab and adds a "noopener" rel to them var stickyLinks = document.querySelectorAll('.grid-item.sticky a'); for (var i = 0; i < stickyLinks.length; i++) { /* stickyLinks[i].setAttribute('target', '_blank'); stickyLinks[i].setAttribute('rel', 'noopener'); */ } // Editorial 302 links, same here var editLinks = document.querySelectorAll( '.grid-item.category-editorial a' ); for (var i = 0; i < editLinks.length; i++) { editLinks[i].setAttribute('target', '_blank'); editLinks[i].setAttribute('rel', 'noopener'); } } // Add current year to copyright notices document.getElementById( 'js-current-year' ).textContent = new Date().getFullYear(); // Call functions after page load extLink(); //addAlly(); setTimeout(function() { //addAlly(); //ideally we should only need to run it once ↑ }, 5000); }; function closeCookieDialog (){ document.getElementById("cookie-consent").style.display = "none"; return false; } setTimeout ( function () { closeCookieDialog(); }, 15000); console.log(`%c 🚀🚀🚀 ███ █████ ███████ █████████ ███████████ █████████████ ███████████████ ███████ ███████ ███████ ┌───────────────────────────────────────────────────────────────────┐ │ │ │ Towards AI is looking for contributors! │ │ Join us in creating awesome AI content. │ │ Let's build the future of AI together → │ │ https://towardsai.net/contribute │ │ │ └───────────────────────────────────────────────────────────────────┘ `, `background: ; color: #00adff; font-size: large`); //Remove latest category across site document.querySelectorAll('a[rel="category tag"]').forEach(function(el) { if (el.textContent.trim() === 'Latest') { // Remove the two consecutive spaces (  ) if (el.nextSibling && el.nextSibling.nodeValue.includes('\u00A0\u00A0')) { el.nextSibling.nodeValue = ''; // Remove the spaces } el.style.display = 'none'; // Hide the element } }); // Add cross-domain measurement, anonymize IPs 'use strict'; //var ga = gtag; ga('config', 'G-9D3HKKFV1Q', 'auto', { /*'allowLinker': true,*/ 'anonymize_ip': true/*, 'linker': { 'domains': [ 'medium.com/towards-artificial-intelligence', 'datasets.towardsai.net', 'rss.towardsai.net', 'feed.towardsai.net', 'contribute.towardsai.net', 'members.towardsai.net', 'pub.towardsai.net', 'news.towardsai.net' ] } */ }); ga('send', 'pageview'); -->