91³Ô¹ÏÍø

Watch CBS News

Grapefest returns to Grapevine with wine, rides and roaring '20s flair

The 39th annual Grapefest is underway in historic downtown Grapevine this weekend. The four-day festival includes wine tastings, kid-friendly events, local food and art vendors, and much more along Main Street.

Grapefest features dozens of rides, ranging from thrilling attractions for adults to gentler options for young children.

From live music to grape stomping — and of course, wine — organizers say there's something for the entire family.

Festival embraces a Gatsby theme

It's the largest wine festival in the Southwest. This year's theme is The Grape Gatsby, celebrating the spirit of the Roaring '20s.

Parking options include free and paid

Paid parking is available nearby, along with access via TEXRail. Free parking options include:

  • 702 S. Main St. – B&D Feed Store – FREE (ADA accessible)
  • 255 E. Dallas Rd. – Grapevine Main Parking Garage – FREE for first 90 minutes; $1 for each additional hour. Costs $15 during special events (ADA accessible)
  • 636 S. Main St. – Grapevine Convention & Visitors Bureau – FREE (ADA accessible)
  • 705 S. Main St. – Historic Cotton Belt Depot – FREE; best for Grapevine Vintage Railroad excursions (ADA accessible)

Vendors include custom hatmaker

Grapefest features more than 100 vendors, including Heidi Brasher, owner and artist behind The Branded Brim. She has sold her custom-made hats at the festival for four years and sees several hundred customers annually.

"I just enjoy the people. It is so much fun to just see women try on hats, create their own brand, and create something that's specific to them. I love when a group of girlfriends come and hyping up, and then husbands will come and buy Christmas gifts or birthday gifts for their wives or daughters," Brasher said.

Festival runs through Sunday evening

Grapefest continues until 11 p.m. Saturday and from 11 a.m. to 6 p.m. on Sunday.

View CBS News In
CBS News App Open
Chrome Safari Continue
const link = doc.createElement('link'); link.rel = 'stylesheet'; link.href = '/fly/fly/bundles/cbsnewscontent/css/cmp-banner.min.css?v=50747257b890e014813016b79ece0fb2'; doc.head.appendChild(link); doc.body.innerHTML = CONSENT_MESSAGE; } else { el.insertAdjacentHTML('afterend', CONSENT_MESSAGE); } }); } function hidePrivacyMessage() { // Remove from the main document document.querySelectorAll(`.${CONSENT_MESSAGE_CLASS}`).forEach(el => el.remove()); // Remove from inside any iframes document.querySelectorAll('iframe').forEach(iframe => { const doc = iframe.contentDocument || iframe.contentWindow.document; doc.querySelectorAll(`.${CONSENT_MESSAGE_CLASS}`).forEach(el => el.remove()); }); } function activateGatedScripts() { // Handle both new format (cmp-gated-script) and old OneTrust/Ketch format (optanon-category-4) const gatedScripts = Array.from(document.querySelectorAll('script.cmp-gated-script, script.optanon-category-4')); // Activate scripts sequentially with a small delay to avoid timing issues let delay = 0; gatedScripts.forEach(function(placeholder, index) { setTimeout(function() { // Skip if already processed if (placeholder.hasAttribute('data-cmp-processed')) { return; } placeholder.setAttribute('data-cmp-processed', 'true'); const newScript = document.createElement('script'); newScript.type = 'text/javascript'; // Try new format first (data-cmp-src), then fall back to old format (data-src) const src = placeholder.getAttribute('data-cmp-src') || placeholder.getAttribute('data-src'); if (src) { newScript.src = src; } else if (placeholder.textContent) { // Inline script - just copy the content newScript.textContent = placeholder.textContent; } // Handle new format attributes (data-cmp-attrs) - for both inline and external scripts const attrs = placeholder.getAttribute('data-cmp-attrs'); if (attrs) { const tempDiv = document.createElement('div'); tempDiv.innerHTML = '
<\/div>'; const tempAttrs = tempDiv.firstChild.attributes; for (let i = 0; i < tempAttrs.length; i++) { // For external scripts, allow defer/async. For inline scripts, skip them (not valid) if (src || (tempAttrs[i].name !== 'async' && tempAttrs[i].name !== 'defer')) { newScript.setAttribute(tempAttrs[i].name, tempAttrs[i].value); } } } // Copy other attributes from old OneTrust format for (let i = 0; i < placeholder.attributes.length; i++) { const attr = placeholder.attributes[i]; // Skip attributes we've already handled or don't want to copy if (!['class', 'data-src', 'data-type', 'data-cmp-src', 'data-cmp-attrs', 'data-cmp-processed', 'type', 'async', 'defer', 'src'].includes(attr.name)) { newScript.setAttribute(attr.name, attr.value); } } placeholder.parentNode.replaceChild(newScript, placeholder); // If external script, manually trigger window.onload handlers after it loads // This handles widgets that use window.onload for initialization if (src) { newScript.addEventListener('load', function() { // If page already loaded and script set a new onload handler, trigger it if (document.readyState === 'complete' && window.onload) { const originalOnload = window.onload; window.onload = null; // Clear temporarily to prevent loops originalOnload(); // Execute the handler } }); } }, delay); delay += 500; // 500ms delay between each script to allow full loading }); } cbsoptanon.onScriptsReady(function(cmp) { cmp.ot.targetingAllowed(function(allowed) { if (!allowed) { showPrivacyMessage(); } else { activateGatedScripts(); } }); cmp.ot.awaitInitialConsent(function(consent_model) { cmp.ot.addOnConsentChangedHandler(function() { cmp.ot.targetingAllowed(function(allowed) { if (allowed) { hidePrivacyMessage(); activateGatedScripts(); } else { showPrivacyMessage(); } }); }); }); });