Class: Www::ShowcasesController

Inherits:
BasePortalController show all
Includes:
Controllers::Paginateable
Defined in:
app/controllers/www/showcases_controller.rb

Constant Summary

Constants included from Controllers::MasqueradeGuarded

Controllers::MasqueradeGuarded::DEFAULT_BLOCK_MESSAGE

Constants included from Controllers::AnalyticsEvents

Controllers::AnalyticsEvents::MAX_QUEUED_EVENTS, Controllers::AnalyticsEvents::SESSION_KEY

Constants included from Controllers::ErrorRendering

Controllers::ErrorRendering::NON_CONTENT_PATH_PREFIXES

Constants included from SeoHelper

Www::SeoHelper::AWARDS, Www::SeoHelper::CA_ADDRESS, Www::SeoHelper::CA_BUSINESS_HOURS, Www::SeoHelper::CA_CONTACT_POINT, Www::SeoHelper::CA_CURRENCIES, Www::SeoHelper::CA_DESCRIPTION, Www::SeoHelper::CA_FOUNDING_DATE, Www::SeoHelper::CA_GLOBAL_LOCATION_NUMBER, Www::SeoHelper::CA_LEGAL_NAME, Www::SeoHelper::CA_LOCAL_BUSINESS, Www::SeoHelper::CA_ONLINE_STORE, Www::SeoHelper::CA_RETURN_POLICY, Www::SeoHelper::CA_SALES_DEPARTMENT, Www::SeoHelper::CA_SERVICE_AREA, Www::SeoHelper::CA_URL, Www::SeoHelper::CA_VAT_ID, Www::SeoHelper::CA_WAREHOUSE_DEPARTMENT, Www::SeoHelper::CA_WAREHOUSE_HOURS, Www::SeoHelper::COMPANY_EMAIL, Www::SeoHelper::COMPANY_LOGO, Www::SeoHelper::COMPANY_NAME, Www::SeoHelper::COMPANY_SLOGAN, Www::SeoHelper::EXPERTISE, Www::SeoHelper::FAX_NUMBER, Www::SeoHelper::GS1_COMPANY_PREFIX, Www::SeoHelper::ISO6523_CODE, Www::SeoHelper::PAYMENT_METHODS, Www::SeoHelper::PHONE_NUMBER, Www::SeoHelper::PRIMARY_NAICS, Www::SeoHelper::REFUND_TYPE, Www::SeoHelper::RETURN_FEES, Www::SeoHelper::RETURN_METHOD, Www::SeoHelper::RETURN_POLICY_CATEGORY, Www::SeoHelper::SECONDARY_NAICS, Www::SeoHelper::SOCIAL_PROFILES, Www::SeoHelper::US_ADDRESS, Www::SeoHelper::US_BUSINESS_HOURS, Www::SeoHelper::US_CONTACT_POINT, Www::SeoHelper::US_CURRENCIES, Www::SeoHelper::US_DESCRIPTION, Www::SeoHelper::US_FOUNDING_DATE, Www::SeoHelper::US_GLOBAL_LOCATION_NUMBER, Www::SeoHelper::US_IMAGE, Www::SeoHelper::US_LEGAL_NAME, Www::SeoHelper::US_LOCAL_BUSINESS, Www::SeoHelper::US_ONLINE_STORE, Www::SeoHelper::US_RETURN_POLICY, Www::SeoHelper::US_SALES_DEPARTMENT, Www::SeoHelper::US_SERVICE_AREA, Www::SeoHelper::US_TAX_ID, Www::SeoHelper::US_URL, Www::SeoHelper::US_WAREHOUSE_DEPARTMENT, Www::SeoHelper::US_WAREHOUSE_HOURS

Constants included from IconHelper

IconHelper::CUSTOM_ICON_MAP, IconHelper::CUSTOM_SVG_DIR, IconHelper::DEFAULT_FAMILY

Instance Method Summary collapse

Methods inherited from BasePortalController

#current_ability, #portal_party, #set_catalog, #set_webpack

Methods included from Controllers::MasqueradeGuarded

block_while_masquerading, #masquerade_blocks?

Methods inherited from ApplicationController

#account_impersonated?, #add_to_flash, #append_token, #bypass_forgery_protection?, #chat_enabled?, #cloudflare_cleared?, #default_catalog, #default_url_options, #enable_turbo_frames, #find_publication, #fix_invalid_accept_header, #init_js_utils, #is_globals_call?, #layout_by_resource, #locale_store, #redirect_to, #require_employee_for_crm, #set_base_host, #set_real_ip, #set_report_errors_for, #should_render_layout?, #stamp_impersonation_context, #warmlyyours_canada_ip?, #warmlyyours_ip?, #y

Methods included from Controllers::ReturnPathHandling

#check_for_return_path, #redirect_to_return_path_or_default

Methods included from Controllers::AnalyticsEvents

#consume_queued_analytics_events, #track_event

Methods included from Controllers::DeviceDetection

#device_detector, #is_ie?

Methods included from Controllers::SubdomainDetection

#is_crm_request?, #is_www_request?, #json_request?

Methods included from Controllers::TrackingDetection

#bot_request?, #gdpr_country?, #gdpr_country_data, #prevent_bots, #set_tracking_cookie, #track_visitor?

Methods included from Controllers::AcceleratedFileSending

#send_file_accelerated, #send_upload_accelerated

Methods included from Controllers::ErrorRendering

#excp_string, #mail_to_for_error_reporting, #render_400, #render_404, #render_406, #render_410, #render_500, #render_invalid_authenticity_token, #render_ip_spoof_error, #safe_referer_or_fallback

Methods included from Controllers::TurnstileVerification

#load_turnstile_script_tag, #turnstile_lazy_widget, #turnstile_script_tag, #turnstile_widget, #validate_turnstile!

Methods included from Controllers::CloudflareCaching

edge_cached, #edge_cached_action?, #reset_cloudflare_cache, #set_cloudflare_cache, #skip_session

Methods included from Controllers::Webpackable

#preload_webpack_fonts, #webpack_css_include, #webpack_css_url, #webpack_js_include, #wpd_is_running?

Methods included from Controllers::Localizable

#cloudflare_country_locale, #determine_request_locale, #geocoder_locale, #guest_user_locale_check, #locale_optional_www_auth_path?, #param_locale, #set_locale, #set_request_locale, #skip_localization?, #warmlyyours_ip_locale

Methods included from Controllers::Authenticable

#access_denied, #authenticate_account, #authenticate_account!, #authenticate_account_from_login_token!, #authenticate_account_from_token!, #check_is_a_manager, #check_is_a_sales_manager, #check_is_an_admin, #check_is_an_employee, #check_party, #clear_mismatched_guest_user, #create_guest_user, #credentials?, #current_or_guest_user, #current_or_guest_user_id_read_only, #current_user, #devise_mapping, #fully_logged_in?, #generate_bot_id, #guest_user, #identifiable?, #init_current_user, #initialize_guest, #load_context_user, #logging_in, #resource, #resource_name, #restrict_access_for_non_employees, #scrubbed_request_path, #user_object, #warn_on_session_guest_id_leak

Methods included from ApplicationHelper

#better_number_to_currency, #check_force_logout, #check_or_cross, #check_or_times, #error_messages, #general_disclaimer_on_product_installation_and_local_codes, #gridjs_from_html_table, #gridjs_table, #is_wy_ip, #line_break, #parent_layout, #pass_or_fail, #render_error_messages_list, #render_video_card, #resolved_auth_form_turbo_frame, #return_path_or, #safe_css_color, #set_return_path_if_present, #set_section_if_present, #tab_frame_id, #to_underscore, #track_page?, #turbo_section_wrapper, #turbo_tabs_request?, #url_on_same_domain_as_request, #widget_index_daily_focus_index_path, #working_hours?, #yes_or_no, #yes_or_no_highlighted, #yes_or_no_with_check_or_cross, #youtube_video

Methods included from UppyUploaderHelper

#file_uploader, #image_uploader, #large_file_uploader_s3, #lead_sketch_uploader, #rma_image_uploader, #rma_image_uploader_s3, #uppy_uploader, #video_uploader

Methods included from ImagesHelper

#image_asset_tag, #image_asset_url

Methods included from SeoHelper

#add_page_schema, #canada?, #company_social_links, #ensure_context_json, #json_ld_script_tag, #local_business_schema, #online_store_id, #online_store_schema, #page_main_entity, #page_main_entity_json, #render_auto_collection_page_schema, #render_collection_page_schema, #render_local_business_schema, #render_online_store_schema, #render_page_schemas, #render_page_video_schemas, #render_webpage_schema, #render_webpage_schema_with_collections, #usa?

Methods included from UrlsHelper

#catalog_breadcrumb_links, #catalog_link, #catalog_link_for_product_line, #catalog_link_for_sku, #cms_link, #delocalized_path, #path_to_sales_product_sku, #path_to_sales_product_sku_for_product_line, #path_to_sales_product_sku_for_product_line_slug, #product_line_from_catalog_link, #protocol_neutral_url, #sanitize_external_url, #valid_external_url?

Methods included from IconHelper

#account_nav_icon, #fa_icon, #star_rating_html

Instance Method Details

#by_room_typeObject

GET /showcases/:room_type

Raises:

  • (ActiveRecord::RecordNotFound)


9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
# File 'app/controllers/www/showcases_controller.rb', line 9

def by_room_type
  slug = params[:room_type].to_s
  @room_type = RoomType.find_by(seo_key: slug) || RoomType.find_by("LOWER(REPLACE(name, ' ', '-')) = ?", slug.downcase)
  raise ActiveRecord::RecordNotFound unless @room_type

  display_name = (@room_type.friendly_name.presence || @room_type.name).to_s
  proper_name = display_name.titleize

  @title = proper_name + ' Floor Plans'
  @meta_description = @description = seo_description_for_room_type(@room_type)
  @breadcrumb = [
    { name: 'Floor Plans', url: cms_link('/floor-plans') },
    { name: proper_name }
  ]

  @showcases = Showcase.published.room_types_include(@room_type.id).order_by_images.order(:name)
  render 'www/showcases/by_room_type'
end

#indexObject

GET /showcases



29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
# File 'app/controllers/www/showcases_controller.rb', line 29

def index
  @title = 'Project Showcase | WarmlyYours'
  @meta_description = 'Explore radiant heating project showcases with filters by floor type, room type, and location.'

  @floor_types = FloorType.order(:name) # retained, but filter removed in UI
  @room_types = RoomType.order(:name)
  @countries = Country.countries_for_select
  @states = State.states_for_select
  # Use static list to replace legacy ShowcaseRoom.product_type_options
  @product_types = [
    ['Floor Heating', 'floor-heating'],
    ['Snow Melting', 'snow-melting'],
    ['Roof & Gutter Deicing', 'roof-gutter'],
    ['Pipe Freeze Protection', 'pipe-freeze'],
    ['LED Mirror', 'led-mirror'],
    ['Towel Warmer', 'towel-warmer'],
    ['Radiant Panel', 'radiant-panel'],
    ['Mirror Defogger', 'mirror-defogger'],
    ['Shower Kit', 'shower-kit']
  ]

  # Normalize URL slug to internal product_type value for dropdown preselection
  # e.g., "/roof-and-gutter-deicing/showcases" → "roof-gutter"
  @selected_product_type = case params[:product_type].to_s.strip
                           when 'roof-and-gutter-deicing' then 'roof-gutter'
                           when 'pipe-freeze-protection' then 'pipe-freeze'
                           when 'shower-kits' then 'shower-kit'
                           else params[:product_type]
                           end

  @q = Showcase.published.ransack(params[:q])
  @showcases = @q.result

  # Room type filter: supports id or slug in params[:room_type]
  if params[:room_type_id].present?
    @showcases = @showcases.room_types_include(params[:room_type_id])
  elsif params[:room_type].present?
    slug = params[:room_type].to_s
    # Use single quotes inside SQL to avoid interpreting as identifiers
    if (rt = RoomType.find_by(seo_key: slug) || RoomType.find_by("LOWER(REPLACE(name, ' ', '-')) = ?", slug.downcase))
      @showcases = @showcases.room_types_include(rt.id)
    end
  end
  @showcases = @showcases.where(state_code: params[:state_code]) if params[:state_code].present?
  @showcases = @showcases.where('city ILIKE ?', "%#{params[:city].to_s.strip}%") if params[:city].present?
  if params[:product_type].present?
    # Map legacy/product UI slugs to canonical ProductLine URLs when needed
    raw_slug = params[:product_type].to_s.strip
    slug_map = {
      'roof-gutter' => 'roof-and-gutter-deicing',
      'pipe-freeze' => 'pipe-freeze-protection',
      'shower-kit' => 'shower-kits'
    }
    canonical_slug = slug_map.fetch(raw_slug, raw_slug)

    product_line_ids = ProductLine.by_url_with_descendants(canonical_slug).pluck(:id).compact

    if product_line_ids.any?
      # Find items that belong to any of the selected product lines (including descendants)
      item_ids = Item.where(primary_product_line_id: product_line_ids).pluck(:id).compact

      @showcases = @showcases.where(
        '(array_remove(product_line_ids, NULL) && ARRAY[?]::integer[]) OR (array_remove(item_ids, NULL) && ARRAY[?]::integer[])',
        product_line_ids, item_ids
      )
    else
      # Fallback to a broad name match if URL mapping didn't resolve anything
      term = raw_slug.tr('-', ' ').strip
      if term.present?
        like = "%#{ActiveRecord::Base.sanitize_sql_like(term)}%"
        @showcases = @showcases.where(
          <<~SQL.squish, like, like
            EXISTS (
              SELECT 1 FROM product_lines pl
              WHERE pl.id = ANY(showcases.product_line_ids) AND pl.name ILIKE ?
            )
            OR EXISTS (
              SELECT 1 FROM items i
              JOIN product_lines pl2 ON pl2.id = i.primary_product_line_id
              WHERE i.id = ANY(showcases.item_ids) AND pl2.name ILIKE ?
            )
          SQL
        )
      end
    end
  end
  if params[:sku].present?
    skus = params[:sku].to_s.upcase.split(/[\s,]+/).map(&:strip).reject(&:blank?)
    if skus.any?
      items = Item.where(sku: skus).pluck(:id).compact
      @showcases = if items.any?
                     @showcases.where('array_remove(item_ids, NULL) && ARRAY[?]::integer[]', items)
                   else
                     @showcases.none
                   end
    end
  end
  if params[:country_iso3].present?
    # Assuming US/CA only — if we store country, filter here; otherwise, infer from state
  end

  @pagy, @showcases = pagy(@showcases.order_by_images.order(created_at: :desc), limit: 24)

  if turbo_frame_request? && params[:page].present?
    render partial: 'next_page', layout: false
    return
  end
end

#showObject

GET /showcases/:slug



139
140
141
# File 'app/controllers/www/showcases_controller.rb', line 139

def show
  set_cloudflare_cache(time_in_secs: 30.days.to_i, tags: %w[showcase])
end