Module: AmazonProductsHelper
- Defined in:
- app/helpers/amazon_products_helper.rb
Overview
View helper: amazon products.
Constant Summary collapse
- AMAZON_SELLER_IDS =
Amazon's seller IDs (US and Canada) - we don't compete with them on price
%w[A2R2RITDJNW1Q6 A3DWYIK6Y9EEQB].freeze
- EXTERNAL_COMPETITOR_ID =
Special marker for external competitors (Amazon threshold-based)
'EXTERNAL_COMPETITOR'
Instance Method Summary collapse
- #build_price_history_html(_catalog_item) ⇒ Object
- #buy_box_indicator(catalog_item) ⇒ Object
-
#cached_amazon_maximum_seller_allowed_price_with_tax(catalog_item) ⇒ Object
Cached maximum seller allowed price calculation (avoids N+1).
-
#cached_competitor_name(seller_id) ⇒ Object
Cached competitor name lookup (uses @amazon_competitors_cache from controller).
-
#cached_edi_log(catalog_item) ⇒ Object
Get cached EDI log for a catalog item (preloaded in controller).
-
#cached_merchant_id(catalog_item) ⇒ Object
Get cached merchant_id (preloaded in controller).
-
#cached_msrp(catalog_item) ⇒ Object
Cached MSRP lookup (avoids N+1 on root_catalog_item).
-
#cached_root_catalog_item(catalog_item) ⇒ Object
Cached root catalog item lookup (uses @root_catalog_items_cache from controller).
-
#cached_sibling_retailer_prices(catalog_item) ⇒ Object
Get sibling catalog items with retail prices from cache (preloaded in controller) These are prices from other retailers (Home Depot, Wayfair, Costco, etc.) for the same item Falls back to direct query when cache is not available (e.g., Turbo Stream updates).
-
#competitive_price_column(catalog_item) ⇒ Object
Competitive price column (dedicated column version).
-
#competitive_price_indicator(catalog_item) ⇒ Object
Competitive price indicator for non-buy-box winners (inline version).
-
#competitor_action_data(catalog_item) ⇒ Object
Returns competitor data for Actions column link.
-
#competitor_storefront_url_for(catalog_item, seller_id) ⇒ Object
Build competitor storefront URL based on catalog marketplace.
-
#data_freshness_indicator(catalog_item) ⇒ Object
Show informational indicator for Buy Box winners with next best competitor price Get next best competitor (not us) from EDI buy box status.
- #featured_merchant_indicator(catalog_item) ⇒ Object
-
#fetch_sibling_retailer_prices_for_item(item_id, catalog_item) ⇒ Object
Fetch sibling retailer prices for a single item (used when cache is not available).
-
#get_best_competitor_from_edi(catalog_item) ⇒ Object
Get the best (lowest price) competitor offer from EDI buy box status logs Returns offers from sellers that are NOT us (checks by SellerId) Also checks external competitor thresholds (CompetitivePriceThreshold, SuggestedLowerPricePlusShipping) Returns: { price: Float, seller_id: String, is_external: Boolean, is_warning: Boolean, is_amazon: Boolean } or nil.
-
#get_competitive_data_from_edi(catalog_item) ⇒ Object
Legacy method for backward compatibility - returns competitive data only when we're NOT winning.
-
#get_winning_fulfillment_type(catalog_item) ⇒ Object
Determine if the winning buy box offer is FBA or MFN.
- #issues_indicator(catalog_item) ⇒ Object
- #market_flag_for(catalog) ⇒ Object
- #price_change_class(catalog_item) ⇒ Object
- #price_change_indicator(catalog_item) ⇒ Object
-
#price_updated_indicator(catalog_item) ⇒ Object
Price updated column with arrow indicator.
-
#price_with_history(catalog_item) ⇒ Object
Price cell with history popover.
- #render_amazon_status_icons(catalog_item) ⇒ Object
- #render_stat_card(label, value, icon, color, url: nil) ⇒ Object
-
#resolve_store_id_for_sibling_filter(catalog_item) ⇒ Object
Resolve the store_id for a catalog item to scope sibling prices to the same country.
-
#retailer_prices_dropdown(catalog_item) ⇒ Object
Render a dropdown showing external retailer prices for the same product Returns nil if no sibling prices are available.
- #row_class_for(catalog_item) ⇒ Object
- #state_indicator(catalog_item) ⇒ Object
- #suppression_indicator(catalog_item) ⇒ Object
Instance Method Details
#build_price_history_html(_catalog_item) ⇒ Object
413 414 415 416 417 |
# File 'app/helpers/amazon_products_helper.rb', line 413 def build_price_history_html(_catalog_item) # PERFORMANCE: Disabled for now - querying versions table per-row is expensive # TODO: Re-enable with batch preloading if price history is needed nil end |
#buy_box_indicator(catalog_item) ⇒ Object
178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 |
# File 'app/helpers/amazon_products_helper.rb', line 178 def buy_box_indicator(catalog_item) repricing_disabled = catalog_item.respond_to?(:disable_amz_repricing) && catalog_item.disable_amz_repricing if catalog_item.is_amz_buy_box_winner # Check if it's FBA or MFN winning fulfillment_type = get_winning_fulfillment_type(catalog_item) if fulfillment_type == :fba tag.span(class: 'status-icon warning', title: 'FBA Buy Box Winner', data: { bs_toggle: 'tooltip' }) do fa_icon('trophy', family: :solid) end else # MFN/FBM winner (or unknown - default to standard) tag.span(class: 'status-icon success', title: 'Buy Box Winner', data: { bs_toggle: 'tooltip' }) do fa_icon('trophy', family: :solid) end end elsif repricing_disabled tag.span(class: 'status-icon warning', title: 'Not winning — repricing disabled (channel protection)', data: { bs_toggle: 'tooltip' }) do fa_icon('pause', family: :solid) end else tag.span(class: 'status-icon muted', title: 'Not Buy Box Winner', data: { bs_toggle: 'tooltip' }) do fa_icon('circle', family: :regular) end end end |
#cached_amazon_maximum_seller_allowed_price_with_tax(catalog_item) ⇒ Object
Cached maximum seller allowed price calculation (avoids N+1)
37 38 39 40 41 42 43 44 |
# File 'app/helpers/amazon_products_helper.rb', line 37 def cached_amazon_maximum_seller_allowed_price_with_tax(catalog_item) msrp = cached_msrp(catalog_item) price = [msrp, catalog_item.amount].compact.max return price unless catalog_item.tax_rate ((catalog_item.tax_rate + 1) * (price || 0.0)).round(2) end |
#cached_competitor_name(seller_id) ⇒ Object
Cached competitor name lookup (uses @amazon_competitors_cache from controller)
9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
# File 'app/helpers/amazon_products_helper.rb', line 9 def cached_competitor_name(seller_id) return nil if seller_id.blank? # Handle external competitor marker return 'External (Amazon Threshold)' if seller_id == EXTERNAL_COMPETITOR_ID if @amazon_competitors_cache.present? competitor = @amazon_competitors_cache[seller_id] return competitor&.display_name || seller_id end # Fallback to database lookup AmazonCompetitor.name_for(seller_id) || seller_id end |
#cached_edi_log(catalog_item) ⇒ Object
Get cached EDI log for a catalog item (preloaded in controller)
207 208 209 210 211 212 213 214 215 216 217 218 |
# File 'app/helpers/amazon_products_helper.rb', line 207 def cached_edi_log(catalog_item) if @edi_logs_cache.present? # Try both integer and string keys (PostgreSQL may return either) return @edi_logs_cache[catalog_item.id] || @edi_logs_cache[catalog_item.id.to_s] end # Fallback to query if not preloaded catalog_item.edi_communication_logs .where(category: :buy_box_status, state: 'processed') .order(created_at: :desc) .first end |
#cached_merchant_id(catalog_item) ⇒ Object
Get cached merchant_id (preloaded in controller)
221 222 223 224 225 226 |
# File 'app/helpers/amazon_products_helper.rb', line 221 def cached_merchant_id(catalog_item) return @merchant_id if @merchant_id.present? # Fallback to query if not preloaded catalog_item.catalog.load_orchestrator&.merchant_id end |
#cached_msrp(catalog_item) ⇒ Object
Cached MSRP lookup (avoids N+1 on root_catalog_item)
32 33 34 |
# File 'app/helpers/amazon_products_helper.rb', line 32 def cached_msrp(catalog_item) cached_root_catalog_item(catalog_item).try(:amount) end |
#cached_root_catalog_item(catalog_item) ⇒ Object
Cached root catalog item lookup (uses @root_catalog_items_cache from controller)
25 26 27 28 29 |
# File 'app/helpers/amazon_products_helper.rb', line 25 def cached_root_catalog_item(catalog_item) return catalog_item if @root_catalog_items_cache.nil? @root_catalog_items_cache[catalog_item.store_item_id] || catalog_item end |
#cached_sibling_retailer_prices(catalog_item) ⇒ Object
Get sibling catalog items with retail prices from cache (preloaded in controller)
These are prices from other retailers (Home Depot, Wayfair, Costco, etc.) for the same item
Falls back to direct query when cache is not available (e.g., Turbo Stream updates)
711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 |
# File 'app/helpers/amazon_products_helper.rb', line 711 def cached_sibling_retailer_prices(catalog_item) item_id = catalog_item.respond_to?(:item_id) ? catalog_item.item_id : catalog_item.store_item&.item_id return [] unless item_id if @sibling_retailer_prices_cache.present? siblings = @sibling_retailer_prices_cache[item_id] || [] store_id = resolve_store_id_for_sibling_filter(catalog_item) return siblings.select { |s| s.store_id == store_id } if store_id return siblings end # Fallback: Direct query for Turbo Stream updates # This is less efficient but only happens for single-item updates fetch_sibling_retailer_prices_for_item(item_id, catalog_item) end |
#competitive_price_column(catalog_item) ⇒ Object
Competitive price column (dedicated column version)
444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 |
# File 'app/helpers/amazon_products_helper.rb', line 444 def competitive_price_column(catalog_item) # Get the best competitor offer from EDI data (any seller that isn't us) edi_data = get_best_competitor_from_edi(catalog_item) return tag.span('—', class: 'text-muted') if edi_data.blank? comp_price = edi_data[:price] competitor_seller_id = edi_data[:seller_id] is_external = edi_data[:is_external] == true is_warning = edi_data[:is_warning] == true # Buy Box winner with external threshold below our price is_amazon_competitor = edi_data[:is_amazon] == true # Amazon is the Buy Box winner return tag.span('—', class: 'text-muted') if comp_price.blank? our_price = catalog_item.amazon_price_with_tax return tag.span('—', class: 'text-muted') if our_price.blank? || our_price.zero? price_diff = our_price - comp_price # Positive means we're higher, negative means we're lower price_diff_pct = (price_diff / comp_price * 100).round(1) # Color coding based on price comparison # Special case: External threshold warning for Buy Box winners → always orange # Green: We beat competitor by 10%+ (our price is 10%+ lower) # Orange: We beat competitor by less than 10% (including same price), OR warning for BB winners # Red: Competitor beats us (their price is lower) and we're NOT the Buy Box winner # Blue: Amazon competitors (we don't compete with them) diff_class = if is_warning 'text-warning' # Buy Box winner but external threshold below our price - warning elsif price_diff.negative? && price_diff_pct <= -10 'text-success' # We're 10%+ cheaper - excellent elsif price_diff.negative? || price_diff.zero? 'text-warning' # We're cheaper by less than 10% OR same price else 'text-danger' # Competitor is cheaper - we need to lower price end # Build popover with competitor info (no links - they don't work in popovers) competitor_name = if is_external 'External' elsif competitor_seller_id.present? cached_competitor_name(competitor_seller_id) else 'Unknown' end popover_html = "<strong>Competitor:</strong> #{h(competitor_name)}" if is_external && is_warning popover_html += " <span class='badge bg-warning text-dark'>Price Ceiling</span>" elsif is_external popover_html += " <span class='badge bg-danger'>Amazon Threshold</span>" elsif is_amazon_competitor popover_html += " <span class='badge' style='background-color: #0d6efd;'>Amazon</span>" end popover_html += "<br><strong>Their price:</strong> #{number_to_currency(comp_price)}" popover_html += "<br><strong>Our price:</strong> #{number_to_currency(our_price)}" # Use inline style for Amazon to ensure blue color renders in popover if is_external popover_html += "<br><strong>Difference:</strong> <span class='#{diff_class}'>#{'+' unless price_diff.negative?}#{number_to_currency(price_diff)} (#{'+' unless price_diff.negative?}#{price_diff_pct}%)</span>" popover_html += if is_warning "<br><em class='text-warning small'>#{fa_icon('circle-exclamation', family: :solid, class: 'me-1')}Price ceiling - don't raise above this</em>" elsif price_diff.positive? "<br><em class='text-danger small'>#{fa_icon('triangle-exclamation', family: :solid, class: 'me-1')}External competitor beating us</em>" else "<br><em class='text-muted small'>External competitor (Amazon threshold)</em>" end elsif is_amazon_competitor popover_html += "<br><strong>Difference:</strong> <span style='color: #0d6efd;'>#{'+' unless price_diff.negative?}#{number_to_currency(price_diff)} (#{'+' unless price_diff.negative?}#{price_diff_pct}%)</span>" popover_html += "<br><em style='color: #0d6efd;' class='small'>#{fa_icon('amazon', family: :brands, class: 'me-1')}We don't auto-compete with Amazon</em>" else popover_html += "<br><strong>Difference:</strong> <span class='#{diff_class}'>#{'+' unless price_diff.negative?}#{number_to_currency(price_diff)} (#{'+' unless price_diff.negative?}#{price_diff_pct}%)</span>" end # Display with optional indicator # External competitor warning (Buy Box winner): show ceiling icon in orange # External competitor danger (not winning): show warning icon in red price_display = if is_external && is_warning safe_join([fa_icon('arrow-up', family: 'solid', class: 'me-1'), number_to_currency(comp_price)]) elsif is_external && price_diff.positive? safe_join([fa_icon('triangle-exclamation', family: 'solid', class: 'me-1'), number_to_currency(comp_price)]) elsif is_external safe_join([tag.span('EXT', class: 'badge bg-secondary me-1', style: 'font-size: 0.6rem;'), number_to_currency(comp_price)]) elsif is_amazon_competitor safe_join([fa_icon('amazon', family: 'brands', class: 'me-1'), number_to_currency(comp_price)]) else number_to_currency(comp_price) end # Use blue for Amazon competitors, red for external display_style = if is_amazon_competitor 'color: #0d6efd !important;' elsif is_external nil # Use text-danger class end tag.span( class: "#{diff_class} cursor-pointer", style: display_style, data: { bs_toggle: 'popover', bs_trigger: 'hover focus', bs_html: 'true', bs_content: popover_html, bs_placement: 'left' } ) do price_display end end |
#competitive_price_indicator(catalog_item) ⇒ Object
Competitive price indicator for non-buy-box winners (inline version)
420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 |
# File 'app/helpers/amazon_products_helper.rb', line 420 def competitive_price_indicator(catalog_item) return '' if catalog_item.is_amz_buy_box_winner # Check for recent flags with competitive price data flag = catalog_item.amazon_catalog_item_flags .where.not(competitive_price: nil) .order(created_at: :desc) .first return '' if flag&.competitive_price.blank? price_diff = flag.competitive_price - catalog_item.amazon_price_with_tax diff_class = price_diff.negative? ? 'text-success' : 'text-danger' tag.small(class: "#{diff_class} d-block") do safe_join([ 'Comp: ', number_to_currency(flag.competitive_price), " (#{'+' if price_diff.positive?}#{number_to_currency(price_diff)})" ]) end end |
#competitor_action_data(catalog_item) ⇒ Object
Returns competitor data for Actions column link
555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 |
# File 'app/helpers/amazon_products_helper.rb', line 555 def competitor_action_data(catalog_item) edi_data = get_best_competitor_from_edi(catalog_item) return nil if edi_data.blank? || edi_data[:seller_id].blank? seller_id = edi_data[:seller_id] # Handle external competitors (Amazon threshold-based) if seller_id == EXTERNAL_COMPETITOR_ID return { seller_id: seller_id, name: 'External (Amazon Threshold)', url: nil, # No storefront URL for external competitors is_unknown: false, is_external: true } end competitor = AmazonCompetitor.find_by(seller_id: seller_id) if @amazon_competitors_cache.blank? competitor ||= @amazon_competitors_cache&.[](seller_id) { seller_id: seller_id, name: competitor&.display_name || seller_id, url: competitor_storefront_url_for(catalog_item, seller_id), is_unknown: competitor.nil? || competitor.name.blank?, is_external: false } end |
#competitor_storefront_url_for(catalog_item, seller_id) ⇒ Object
Build competitor storefront URL based on catalog marketplace
585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 |
# File 'app/helpers/amazon_products_helper.rb', line 585 def competitor_storefront_url_for(catalog_item, seller_id) # Use marketplace_identifier directly if available (from ViewAmazonCatalogItem) # Otherwise fall back to chaining through associations (CatalogItem) identifier = if catalog_item.respond_to?(:marketplace_identifier) catalog_item.marketplace_identifier else catalog_item.catalog&.amazon_marketplace&.marketplace_identifier end marketplace_domain = case identifier when 'ATVPDKIKX0DER' then 'amazon.com' when 'A2EUQ1WTGCTBG2' then 'amazon.ca' when 'A1F83G8C2ARO7P' then 'amazon.co.uk' when 'A1PA6795UKMFR9' then 'amazon.de' when 'A13V1IB3VIYZZH' then 'amazon.fr' when 'APJ6JRA9NG5V4' then 'amazon.it' when 'A1RKKUPIHCS9HS' then 'amazon.es' else 'amazon.com' end "https://www.#{marketplace_domain}/sp?seller=#{seller_id}" end |
#data_freshness_indicator(catalog_item) ⇒ Object
Show informational indicator for Buy Box winners with next best competitor price
Get next best competitor (not us) from EDI buy box status
332 333 334 335 336 337 338 339 340 341 342 343 344 345 |
# File 'app/helpers/amazon_products_helper.rb', line 332 def data_freshness_indicator(catalog_item) return tag.span('Never', class: 'text-muted') if catalog_item.amazon_info_datetime.blank? time_ago = time_ago_in_words(catalog_item.amazon_info_datetime) freshness_class = if catalog_item.amazon_info_datetime > 24.hours.ago 'text-success' elsif catalog_item.amazon_info_datetime > 7.days.ago 'text-warning' else 'text-danger' end tag.span(time_ago, class: freshness_class, title: catalog_item.amazon_info_datetime.strftime('%Y-%m-%d %H:%M'), data: { bs_toggle: 'tooltip' }) end |
#featured_merchant_indicator(catalog_item) ⇒ Object
249 250 251 252 253 254 255 256 257 258 259 |
# File 'app/helpers/amazon_products_helper.rb', line 249 def featured_merchant_indicator(catalog_item) if catalog_item.is_amz_featured_merchant tag.span(class: 'status-icon warning', title: 'Featured Merchant', data: { bs_toggle: 'tooltip' }) do fa_icon('star', family: :solid) end else tag.span(class: 'status-icon muted', title: 'Not Featured', data: { bs_toggle: 'tooltip' }) do fa_icon('circle', family: :regular) end end end |
#fetch_sibling_retailer_prices_for_item(item_id, catalog_item) ⇒ Object
Fetch sibling retailer prices for a single item (used when cache is not available)
737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 |
# File 'app/helpers/amazon_products_helper.rb', line 737 def fetch_sibling_retailer_prices_for_item(item_id, catalog_item) store_id = catalog_item.respond_to?(:store_id) ? catalog_item.store_id : catalog_item.catalog&.store_id return [] unless store_id CatalogItem .joins(:catalog, :store_item) .where(catalogs: { store_id: store_id, external_price_check_enabled: true }) .where(store_items: { item_id: item_id }) .where(state: 'active') .where.not(retail_price: nil) .where('catalog_items.retail_price > 0') .select( 'catalog_items.id as catalog_item_id', 'catalog_items.retail_price', 'catalog_items.retailer_price_updated_at as retail_price_updated_at', 'catalog_items.url', 'catalogs.name as catalog_name', 'catalogs.currency as retailer_currency' ) .to_a end |
#get_best_competitor_from_edi(catalog_item) ⇒ Object
Get the best (lowest price) competitor offer from EDI buy box status logs
Returns offers from sellers that are NOT us (checks by SellerId)
Also checks external competitor thresholds (CompetitivePriceThreshold, SuggestedLowerPricePlusShipping)
Returns: { price: Float, seller_id: String, is_external: Boolean, is_warning: Boolean, is_amazon: Boolean } or nil
Logic:
- If Amazon is Buy Box winner → show their price in blue (is_amazon: true)
- If we're Buy Box winner and external threshold < our price → show as warning (orange)
- If we're NOT Buy Box winner and external/seller competitor < our price → show as danger (red)
- Seller competitors take priority over external threshold (unless external is lower)
620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 |
# File 'app/helpers/amazon_products_helper.rb', line 620 def get_best_competitor_from_edi(catalog_item) ecl = cached_edi_log(catalog_item) return nil if ecl&.data.blank? json_hash = JSON.parse(ecl.data).with_indifferent_access our_price = catalog_item.amazon_price_with_tax is_buy_box_winner = catalog_item.is_amz_buy_box_winner # Extract external competitor threshold from Summary summary = json_hash.dig(:payload, :Summary) external_threshold = nil if summary.present? competitive_threshold = summary.dig(:CompetitivePriceThreshold, :Amount)&.to_f suggested_lower = summary.dig(:SuggestedLowerPricePlusShipping, :Amount)&.to_f thresholds = [competitive_threshold, suggested_lower].compact.reject(&:zero?) external_threshold = thresholds.min if thresholds.any? end # Get offers from EDI data offers = json_hash.dig(:payload, :Offers) || [] return nil if offers.empty? offers = offers.map(&:with_indifferent_access) merchant_id = cached_merchant_id(catalog_item) # Check if Amazon is the Buy Box winner - if so, show their price in blue amazon_winner = offers.find { |o| AMAZON_SELLER_IDS.include?(o[:SellerId]) && o[:IsBuyBoxWinner] } if amazon_winner.present? price = amazon_winner.dig(:ListingPrice, :Amount) || amazon_winner.dig(:Price, :Amount) return { price: price&.to_f, seller_id: amazon_winner[:SellerId], is_external: false, is_warning: false, is_amazon: true } end # Get all competitor offers (not from us, not from Amazon) competitor_offers = offers.reject do |o| o[:SellerId] == merchant_id || AMAZON_SELLER_IDS.include?(o[:SellerId]) end seller_competitor = nil if competitor_offers.any? # Get the lowest priced competitor lowest_offer = competitor_offers.min_by do |offer| offer.dig(:ListingPrice, :Amount) || offer.dig(:Price, :Amount) || Float::INFINITY end if lowest_offer price = lowest_offer.dig(:ListingPrice, :Amount) || lowest_offer.dig(:Price, :Amount) seller_competitor = { price: price&.to_f, seller_id: lowest_offer[:SellerId], is_external: false, is_warning: false, is_amazon: false } end end # Case 1: We ARE the Buy Box winner # Show external threshold as WARNING (orange) if it's below our price if is_buy_box_winner return { price: external_threshold, seller_id: EXTERNAL_COMPETITOR_ID, is_external: true, is_warning: true, is_amazon: false } if external_threshold.present? && our_price.present? && external_threshold < our_price # We're winning and no threatening external threshold - show seller competitor if any return seller_competitor if seller_competitor.present? return nil end # Case 2: We are NOT the Buy Box winner # Show the lowest threat (seller or external) - this is a danger situation if seller_competitor.present? # We have a seller competitor - only consider external if it's lower than BOTH if external_threshold.present? && external_threshold < seller_competitor[:price] && our_price.present? && external_threshold < our_price return { price: external_threshold, seller_id: EXTERNAL_COMPETITOR_ID, is_external: true, is_warning: false, is_amazon: false } else return seller_competitor end elsif external_threshold.present? && our_price.present? && external_threshold < our_price # No seller competitor, but external threshold is below our price - show it return { price: external_threshold, seller_id: EXTERNAL_COMPETITOR_ID, is_external: true, is_warning: false, is_amazon: false } end nil rescue JSON::ParserError, NoMethodError nil end |
#get_competitive_data_from_edi(catalog_item) ⇒ Object
Legacy method for backward compatibility - returns competitive data only when we're NOT winning
702 703 704 705 706 |
# File 'app/helpers/amazon_products_helper.rb', line 702 def get_competitive_data_from_edi(catalog_item) return nil if catalog_item.is_amz_buy_box_winner get_best_competitor_from_edi(catalog_item) end |
#get_winning_fulfillment_type(catalog_item) ⇒ Object
Determine if the winning buy box offer is FBA or MFN
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 |
# File 'app/helpers/amazon_products_helper.rb', line 229 def get_winning_fulfillment_type(catalog_item) ecl = cached_edi_log(catalog_item) return :unknown if ecl&.data.blank? json_hash = JSON.parse(ecl.data).with_indifferent_access offers = json_hash.dig(:payload, :Offers) || [] return :unknown if offers.empty? offers = offers.map(&:with_indifferent_access) merchant_id = cached_merchant_id(catalog_item) # Find our winning offer our_winning_offer = offers.find { |o| o[:SellerId] == merchant_id && o[:IsBuyBoxWinner] } return :unknown unless our_winning_offer our_winning_offer[:IsFulfilledByAmazon] ? :fba : :mfn rescue JSON::ParserError, NoMethodError :unknown end |
#issues_indicator(catalog_item) ⇒ Object
261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 |
# File 'app/helpers/amazon_products_helper.rb', line 261 def issues_indicator(catalog_item) active_flags = catalog_item.amazon_catalog_item_flags.reject(&:resolved?) repricing_disabled = catalog_item.respond_to?(:disable_amz_repricing) && catalog_item.disable_amz_repricing # No issues - show dash (unless repricing is disabled) return tag.span('—', class: 'text-muted') if active_flags.empty? && !repricing_disabled # Separate flags by type - severity order: red (critical) > orange (warning) # Red flags: has_issues, no_buyable_offers, external_competitor (unexpected, needs attention) # Orange flags: out_of_stock, cannot_automatically_win_buy_box, repricing_disabled (expected/informational) catalog_issues = active_flags.select { |f| f.flag_type == 'has_issues' } no_offers_issues = active_flags.select { |f| f.flag_type == 'no_buyable_offers' } external_competitor_issues = active_flags.select { |f| f.flag_type == 'external_competitor' } issue_parts = [] issue_parts << "<strong>Repricing disabled:</strong><br>Automatic Buy Box repricing paused to protect channel pricing (e.g. Costco)" if repricing_disabled active_flags.each do |flag| content = "<strong>#{flag.flag_type.humanize}:</strong><br>#{h(flag.issues&.truncate(200))}" # Add competitive price info if available if flag.competitive_price.present? content += "<br><small class='text-muted'>Competitive: #{number_to_currency(flag.competitive_price)}</small>" content += "<br><small class='text-muted'>Our price: #{number_to_currency(flag.our_price_at_flagging)}</small>" if flag.our_price_at_flagging.present? end # Add competitor info with link if available (except for external competitors) if flag.competitor_seller_id.present? && flag.competitor_seller_id != EXTERNAL_COMPETITOR_ID competitor_name = flag.competitor_name competitor_url = flag.competitor_storefront_url content += "<br><small class='text-muted'>Competitor: " safe_competitor_url = competitor_url.to_s.match?(%r{\Ahttps?://}i) ? competitor_url : '#' content += "<a href=\"#{ERB::Util.html_escape(safe_competitor_url)}\" target=\"_blank\" class=\"text-decoration-none\" rel=\"noopener noreferrer\">#{h(competitor_name)} <i class=\"fa-solid fa-external-link-alt fa-xs\"></i></a>" content += '</small>' elsif flag.competitor_seller_id == EXTERNAL_COMPETITOR_ID content += "<br><small class='text-muted'>Competitor: External (Amazon Threshold)</small>" end issue_parts << content end issues_html = issue_parts.join('<hr class="my-1">') # Determine icon color based on severity: # - Red (danger): catalog issues, no buyable offers, or external competitor (unexpected, needs attention) # - Orange (warning): out of stock, threshold issues, or repricing disabled (expected/informational) icon_class = if catalog_issues.any? || no_offers_issues.any? || external_competitor_issues.any? 'status-icon danger' # Red for catalog issues, no buyable offers, or external competitor else 'status-icon warning' # Orange for out of stock, threshold issues, or repricing disabled end badge_count = active_flags.count + (repricing_disabled ? 1 : 0) tag.span( class: icon_class, title: '', data: { bs_toggle: 'popover', bs_trigger: 'hover focus', bs_html: 'true', bs_content: issues_html, bs_placement: 'left' } ) do safe_join([ fa_icon('triangle-exclamation', family: :solid), (tag.span(badge_count, class: 'ms-1 small fw-bold') if badge_count > 1) ].compact) end end |
#market_flag_for(catalog) ⇒ Object
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 |
# File 'app/helpers/amazon_products_helper.rb', line 64 def market_flag_for(catalog) case catalog.id when CatalogConstants::AMAZON_SC_US_CATALOG_ID tag.span('🇺🇸', title: 'US', class: 'fs-5') when CatalogConstants::AMAZON_SC_CA_CATALOG_ID tag.span('🇨🇦', title: 'CA', class: 'fs-5') when CatalogConstants::AMAZON_SC_UK_CATALOG_ID tag.span('🇬🇧', title: 'UK', class: 'fs-5') when CatalogConstants::AMAZON_SC_DE_CATALOG_ID tag.span('🇩🇪', title: 'DE', class: 'fs-5') when CatalogConstants::AMAZON_SC_FR_CATALOG_ID tag.span('🇫🇷', title: 'FR', class: 'fs-5') when CatalogConstants::AMAZON_SC_IT_CATALOG_ID tag.span('🇮🇹', title: 'IT', class: 'fs-5') when CatalogConstants::AMAZON_SC_ES_CATALOG_ID tag.span('🇪🇸', title: 'ES', class: 'fs-5') when CatalogConstants::AMAZON_SC_NL_CATALOG_ID tag.span('🇳🇱', title: 'NL', class: 'fs-5') when CatalogConstants::AMAZON_SC_PL_CATALOG_ID tag.span('🇵🇱', title: 'PL', class: 'fs-5') when CatalogConstants::AMAZON_SC_SE_CATALOG_ID tag.span('🇸🇪', title: 'SE', class: 'fs-5') when CatalogConstants::AMAZON_SC_BE_CATALOG_ID tag.span('🇧🇪', title: 'BE', class: 'fs-5') else tag.span('🌐', title: catalog.name, class: 'fs-5') end end |
#price_change_class(catalog_item) ⇒ Object
165 166 167 168 169 170 171 172 173 174 175 176 |
# File 'app/helpers/amazon_products_helper.rb', line 165 def price_change_class(catalog_item) return '' unless catalog_item.price_updated_at.present? && catalog_item.price_updated_at > 7.days.ago return '' unless catalog_item.old_amount.present? && catalog_item.amount.present? if catalog_item.amount > catalog_item.old_amount 'price-up fw-medium' elsif catalog_item.amount < catalog_item.old_amount 'price-down fw-medium' else '' end end |
#price_change_indicator(catalog_item) ⇒ Object
140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 |
# File 'app/helpers/amazon_products_helper.rb', line 140 def price_change_indicator(catalog_item) return '' unless catalog_item.price_updated_at.present? && catalog_item.price_updated_at > 7.days.ago if catalog_item.old_amount.present? && catalog_item.amount.present? if catalog_item.amount > catalog_item.old_amount tag.span(class: 'status-icon success', title: "Price raised on #{catalog_item.price_updated_at.strftime('%b %d')}", data: { bs_toggle: 'tooltip' }) do fa_icon('arrow-up', family: :solid) end elsif catalog_item.amount < catalog_item.old_amount tag.span(class: 'status-icon danger', title: "Price lowered on #{catalog_item.price_updated_at.strftime('%b %d')}", data: { bs_toggle: 'tooltip' }) do fa_icon('arrow-down', family: :solid) end end else '' end end |
#price_updated_indicator(catalog_item) ⇒ Object
Price updated column with arrow indicator
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 |
# File 'app/helpers/amazon_products_helper.rb', line 348 def price_updated_indicator(catalog_item) return tag.span('—', class: 'text-muted') if catalog_item.price_updated_at.blank? time_str = if catalog_item.price_updated_at > 24.hours.ago time_ago_in_words(catalog_item.price_updated_at) else catalog_item.price_updated_at.strftime('%b %d') end arrow = if catalog_item.old_amount.present? && catalog_item.amount.present? if catalog_item.amount > catalog_item.old_amount fa_icon('arrow-trend-up', family: :solid, class: 'text-success me-1') elsif catalog_item.amount < catalog_item.old_amount fa_icon('arrow-trend-down', family: :solid, class: 'text-danger me-1') else '' end else '' end # Determine text color based on recency text_class = if catalog_item.price_updated_at > 7.days.ago 'text-dark' elsif catalog_item.price_updated_at > 30.days.ago 'text-muted' else 'text-muted opacity-50' end tooltip = "Updated: #{catalog_item.price_updated_at.strftime('%Y-%m-%d %H:%M')}" tooltip += "\nOld: #{number_to_currency(catalog_item.old_amount)}" if catalog_item.old_amount.present? tooltip += "\nNew: #{number_to_currency(catalog_item.amount)}" if catalog_item.amount.present? tag.span(class: text_class, title: tooltip, data: { bs_toggle: 'tooltip' }) do safe_join([arrow, time_str].compact) end end |
#price_with_history(catalog_item) ⇒ Object
Price cell with history popover
388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 |
# File 'app/helpers/amazon_products_helper.rb', line 388 def price_with_history(catalog_item) current_price = number_to_currency(catalog_item.amount) # Build price history from PaperTrail versions if available history_html = build_price_history_html(catalog_item) if history_html.present? tag.span( current_price, class: price_change_class(catalog_item), style: 'cursor: pointer;', data: { bs_toggle: 'popover', bs_trigger: 'hover focus', bs_html: 'true', bs_title: 'Price History', bs_content: history_html, bs_placement: 'left' } ) else tag.span(current_price, class: price_change_class(catalog_item)) end end |
#render_amazon_status_icons(catalog_item) ⇒ Object
100 101 102 103 104 105 106 107 108 109 110 111 112 113 |
# File 'app/helpers/amazon_products_helper.rb', line 100 def render_amazon_status_icons(catalog_item) icons = [] # State indicator icons << state_indicator(catalog_item) # Price change indicator (if recently changed) icons << price_change_indicator(catalog_item) # Suppression indicator icons << suppression_indicator(catalog_item) if catalog_item.item_suppressed_status.present? safe_join(icons, ' ') end |
#render_stat_card(label, value, icon, color, url: nil) ⇒ Object
46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 |
# File 'app/helpers/amazon_products_helper.rb', line 46 def render_stat_card(label, value, icon, color, url: nil) card_content = safe_join([ tag.div(class: "stat-value text-#{color}") do safe_join([ fa_icon(icon, class: 'me-2', style: 'font-size: 1rem;'), number_with_delimiter(value) ]) end, tag.div(label, class: 'stat-label') ]) if url.present? link_to(card_content, url, class: 'stat-card stat-card-link text-decoration-none') else tag.div(card_content, class: 'stat-card') end end |
#resolve_store_id_for_sibling_filter(catalog_item) ⇒ Object
Resolve the store_id for a catalog item to scope sibling prices to the same country.
Uses @catalogs_selected (preloaded in controller) to avoid N+1 queries.
730 731 732 733 734 |
# File 'app/helpers/amazon_products_helper.rb', line 730 def resolve_store_id_for_sibling_filter(catalog_item) return catalog_item.store_id if catalog_item.respond_to?(:store_id) && catalog_item.store_id.present? @catalogs_selected&.find { |c| c.id == catalog_item.catalog_id }&.store_id if catalog_item.respond_to?(:catalog_id) && catalog_item.catalog_id end |
#retailer_prices_dropdown(catalog_item) ⇒ Object
Render a dropdown showing external retailer prices for the same product
Returns nil if no sibling prices are available
761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 |
# File 'app/helpers/amazon_products_helper.rb', line 761 def retailer_prices_dropdown(catalog_item) sibling_prices = cached_sibling_retailer_prices(catalog_item) return nil if sibling_prices.blank? our_price = catalog_item.amazon_price_with_tax # Build dropdown items dropdown_items = sibling_prices.map do |sibling| price = sibling.retail_price catalog_name = sibling.catalog_name currency = sibling.retailer_currency || 'USD' url = sibling.url updated_at = sibling.retail_price_updated_at catalog_item_id = sibling.catalog_item_id # Calculate price difference if we have our price diff_html = if our_price.present? && our_price.positive? && price.present? price_diff = our_price - price diff_class = if price_diff.positive? 'text-danger' # We're more expensive elsif price_diff.negative? 'text-success' # We're cheaper else 'text-muted' # Same price end tag.span(class: "#{diff_class} small ms-1") do "(#{'+' unless price_diff.negative?}#{number_to_currency(price_diff, unit: currency == 'CAD' ? 'C$' : '$')})" end end # Format date indicator - show warning color if stale (>7 days old) date_html = if updated_at.present? days_old = (Time.current - updated_at).to_i / 1.day date_class = days_old > 7 ? 'text-warning' : 'text-muted' date_text = if days_old == 0 'today' elsif days_old == 1 '1d ago' elsif days_old < 30 "#{days_old}d ago" else updated_at.strftime('%-m/%-d') end tag.span(class: "#{date_class} small ms-1", title: updated_at.strftime('%Y-%m-%d %H:%M')) do "(#{date_text})" end end # External link icon (opens listing URL in new tab) external_link_html = if url.present? link_to(url, target: '_blank', rel: 'noopener', class: 'text-muted ms-2 d-inline', title: 'Open listing') do fa_icon('up-right-from-square', family: :solid, class: 'fa-xs') end end # Format the dropdown item - retailer name links to CRM catalog item content = tag.div(class: 'd-flex justify-content-between align-items-center px-2 py-1') do # Left side: retailer name (links to CRM catalog item) + external link icon left = tag.span(class: 'me-3 text-nowrap') do retailer_link = link_to(catalog_name, catalog_item_path(catalog_item_id), class: 'text-decoration-none d-inline') safe_join([retailer_link, external_link_html].compact) end # Right side: price, difference, and date right = tag.span(class: 'fw-medium text-nowrap') do safe_join([ number_to_currency(price, unit: currency == 'CAD' ? 'C$' : '$'), diff_html, date_html ].compact) end safe_join([left, right]) end tag.span(content, class: 'dropdown-item small') end # Return the dropdown structure tag.div(class: 'dropdown d-inline-block ms-1') do safe_join([ tag.( class: 'btn btn-link btn-sm p-0 text-decoration-none', type: 'button', data: { bs_toggle: 'dropdown', bs_auto_close: 'true' }, title: 'View retailer prices', 'aria-expanded': 'false' ) do tag.span(class: 'badge bg-info text-white', style: 'font-size: 0.6rem;') do "+#{sibling_prices.count}" end end, tag.ul(class: 'dropdown-menu dropdown-menu-end', style: 'min-width: 300px;') do safe_join([ tag.li(tag.span('Other Retailer Prices', class: 'dropdown-header')), tag.li(tag.hr(class: 'dropdown-divider my-1')), *dropdown_items.map { |item| tag.li(item) } ]) end ]) end end |
#row_class_for(catalog_item) ⇒ Object
93 94 95 96 97 98 |
# File 'app/helpers/amazon_products_helper.rb', line 93 def row_class_for(catalog_item) return 'has-issues' if catalog_item.amazon_catalog_item_flags.any? { |f| !f.resolved? } return 'table-warning' if catalog_item.item_suppressed_status.present? '' end |
#state_indicator(catalog_item) ⇒ Object
115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 |
# File 'app/helpers/amazon_products_helper.rb', line 115 def state_indicator(catalog_item) case catalog_item.state when 'active' tag.span(class: 'status-icon success', title: 'Active', data: { bs_toggle: 'tooltip' }) do fa_icon('check', family: :solid) end when 'active_hidden' tag.span(class: 'status-icon warning', title: 'Active (Hidden)', data: { bs_toggle: 'tooltip' }) do fa_icon('eye-slash', family: :solid) end when 'pending_onboarding' tag.span(class: 'status-icon info', title: 'Pending Onboarding', data: { bs_toggle: 'tooltip' }) do fa_icon('clock', family: :solid) end when 'inactive' tag.span(class: 'status-icon muted', title: 'Inactive', data: { bs_toggle: 'tooltip' }) do fa_icon('pause', family: :solid) end else tag.span(class: 'status-icon muted', title: catalog_item.state.humanize, data: { bs_toggle: 'tooltip' }) do fa_icon('question', family: :solid) end end end |
#suppression_indicator(catalog_item) ⇒ Object
158 159 160 161 162 163 |
# File 'app/helpers/amazon_products_helper.rb', line 158 def suppression_indicator(catalog_item) = catalog_item..presence || catalog_item.item_suppressed_status tag.span(class: 'status-icon danger', title: "Suppressed: #{}", data: { bs_toggle: 'tooltip' }) do fa_icon('ban', family: :solid) end end |