Popular
Meijer Discounts, Deals and Coupon Codes

Grocery store coupon clipping automation (Mperks, Kroger, publix etc..) using JavaScript (similar to the Amex offers Javascript)

4,541 12,362 April 10, 2025 at 11:15 AM in Grocery (5) Meijer
Deal
Score
+34
14,313 Views

Deal Details

Last Edited by Cobalt_Blue_FF August 7, 2025 at 02:18 PM
+34 Deal Score
14,313 Views
Similar to the Amex offers JavaScript that you can bookmark and run to clip all offers, I crafted one that works for Meijer Mperks. Note that Mperks has a limit of 160 coupons clipped per their TOC, so I added a clip limit of 160 to the code . It was not real clear if that is how many you can clip per x amount of time or the total number of clipped coupons.

Either way, I found it threw an error after clipping a bunch of them. But it did work!

Also, here is a direct link to your handpicked offers, it is not always obvious when these are available.

https://www.meijer.com/shopping/c...ffers.html

Mods please delete if this is not allowed to be posted. Thanks!

Just create a new bookmark, and add this code as the webpage. Then click it while on the Amex offers or Mperks coupon pages

Enjoy!

I am adding all new stores to the wiki

Update: See Wiki for most up to date list

Community Wiki

Last Edited by Cobalt_Blue_FF November 17, 2025 at 04:49 AM
JavaScript Clipping Bookmarklets for Grocery & Rewards Sites
  • Mperks Clipper
    Clicks up to 160 coupons with a short delay between each.
    • To change how many coupons it tries to clip, edit: clipLimit=160
    • To adjust the click delay speed, edit: Math.random()*300+100
    Code:
    javascript:(function(){var clipBtns=[...document.querySelectorAll('button.coupon-tile__button--clip[aria-label*="Clip"]')];console.log('Found '+clipBtns.length+' clip buttons');if(clipBtns.length===0){alert('No clip buttons found!');return;}var clipLimit=160;var index=0;function clipNext(){if(index>=clipBtns.length||clipLimit<=0){alert('Clipping complete! Processed '+index+' coupons.');return;}var btn=clipBtns[index];if(btn&&btn.offsetParent!==null){btn.click();console.log('Clipped coupon '+(index+1)+'/'+clipBtns.length);clipLimit--;}index++;setTimeout(clipNext,Math.random()*300+100);}clipNext();})();
  • Mperks Unclipper
    Unclips already clipped coupons. Useful for removing unwanted clipped coupons to filter out deals you don't want and stay under the 160 coupon limit.
    • To change how many coupons it tries to unclip, edit: unclipLimit=160
    • To adjust the timing, edit: Math.random()*200+100 (next coupon delay) or 300 (detail view wait)
    Code:
    javascript:(function(){var unclipBtns=[...document.querySelectorAll('button.coupon-tile__button--unclip[aria-label*="Shop"]')];console.log('Found '+unclipBtns.length+' clipped coupons to unclip');if(unclipBtns.length===0){alert('No clipped coupons found!');return;}var unclipLimit=160;var index=0;function unclipNext(){if(index>=unclipBtns.length||unclipLimit<=0){alert('Unclipping complete! Processed '+index+' coupons.');return;}var btn=unclipBtns[index];if(btn&&btn.offsetParent!==null){btn.click();console.log('Opening coupon detail '+(index+1)+'/'+unclipBtns.length);setTimeout(function(){var unclipDetailBtn=document.querySelector('button.coupon-detail-view__unclip-btn');if(unclipDetailBtn){unclipDetailBtn.click();console.log('Unclipped coupon '+(index+1)+'/'+unclipBtns.length);}else{console.log('Unclip detail button not found for coupon '+(index+1));}unclipLimit--;index++;setTimeout(unclipNext,Math.random()*200+100);},300);}else{index++;unclipNext();}}unclipNext();})();
  • Amex Offers Clipper
    Adds all "Add to Card" offers with a short random delay between each.
    • To adjust delay speed, change the delay function inside the code (currently Math.random()*800+300)
    Code:
    javascript:(function(){
        /* Define multiple selectors to catch the button in various states/versions */
        var selectors = [
            'button[data-testid="merchantOfferListAddButton"]',
            'button[aria-label*="Add to Card" i]',
            'button[title*="Add to Card" i]',
            'button[aria-label*="Activate Offer" i]'
        ];
        
        /* Find all matching buttons that are visible and not disabled */
        var btns = Array.from(document.querySelectorAll(selectors.join(',')))
            .filter(b => b.offsetParent !== null && !b.disabled);
    
        if(!btns.length){
            alert('Amex Clipper: No add-to-list-card buttons found.\n\nMake sure you are on the "Available Offers" tab and have scrolled down to load them.');
            return;
        }
    
        var i=0, limit=btns.length, R=Math.random, delay=()=>R()*800+300;
    
        function realClick(el){
            try{
                ['pointerdown','pointerup','mousedown','mouseup','click'].forEach(type=>{
                    el.dispatchEvent(new MouseEvent(type,{bubbles:true,cancelable:true,view:window}));
                });
            }catch(e){
                try{el.click();}catch(_){}
            }
        }
    
        (function next(){
            if(i>=limit){
                alert('Amex Clipper: finished '+i+' offers');
                return;
            }
            var b=btns[i++];
            /* Check if button state changed while we were waiting (e.g. dupes) */
            if(b.disabled || b.offsetParent === null) {
                next(); 
                return;
            }
            try{
                b.scrollIntoView({block:'center',behavior:'smooth'});
            }catch(_){}
            
            realClick(b);
            setTimeout(next, delay());
        })();
    })();
  • Kroger
    Clicks all "Clip" buttons instantly.
    • No adjustable values.
    Code:
    javascript:[...document.querySelectorAll('button[aria-label*="Clip"]')].forEach(btn => btn.click());
  • Kroger (Without Boost Coupons)
    Same as above, but skips Boost-exclusive coupons.
    • No adjustable values.
    Code:
    javascript:[...document.querySelectorAll('button[aria-label*="Clip"]')].filter(btn => !btn.textContent.includes("Boost")).forEach(btn => btn.click());
  • Publix
    Clicks all coupon buttons instantly.
    • No adjustable values.
    Code:
    javascript:(function(){document.querySelectorAll('.p-coupon-button__label').forEach(button => button.click());})();
  • Winn-Dixie
    Clicks coupons with a small random delay between each.
    • To adjust delay speed, change: Math.random()*300
    Code:
    javascript:btns=[...document.querySelectorAll('span[id^="cpnSpan_"]')];c=()=>{b=btns.shift();if(!b)return;b.click();setTimeout(c,Math.random()*300)};c();
  • H-E-B
    Clips only buttons labeled "Clip" with delay between clicks.
    • To adjust delay speed, change: Math.random()*300
    Code:
    javascript:btns=[...document.querySelectorAll('span.ButtonLabel_buttonLabel__z680Q')].filter(btn=>btn.textContent.trim()==='Clip');c=()=>{b=btns.shift();if(!b)return;b.click();setTimeout(c,Math.random()*300)};c();
  • Kroger Cash Back
    Clips Kroger Cash Back offers with a delay between each.
    • To adjust delay speed, change: Math.random()*300
    Code:
    javascript:btns=[...document.querySelectorAll('button[data-testid^="CouponActionButton-"]')];c=()=>{b=btns.shift();if(!b)return;b.click();setTimeout(c,Math.random()*300)};c();
  • Ibotta
    Clips a limited number of offers with random delays to prevent account lockouts.
    • View your Ibotta items here: https://ibotta.com/retailers/
    • To change how many offers are clipped, edit: limit=5
    • To change the random delay range, edit:
      1250 = randomness (range of delay in ms)
      250 = minimum delay in ms
    Code:
    javascript:(()=>{let btns=[...document.querySelectorAll('span.p-btn__icon.p-btn__icon-middle.p-btn__icon--plus')],limit=5;const c=()=>{if(limit<=0||btns.length===0)return;let b=btns.shift();b.click();limit--;setTimeout(c,Math.random()*1250+250)};c();})()
  • Ralphs
    Clips all coupon buttons with delay to prevent rate limiting. (Uses Kroger infrastructure)
    Code:
    javascript:btns=[...document.querySelectorAll('button, a, [role="button"]')].filter(b => {const text = b.textContent?.trim() || ''; const ariaLabel = b.getAttribute('aria-label') || ''; return text === 'Clip' || text === 'Add Coupon' || ariaLabel.includes('Clip') || ariaLabel.includes('Add Coupon');});c=()=>{ b = btns.shift(); if (!b) return; b.click(); setTimeout(c, Math.random() * 500 + 200) };if(btns.length > 0) c(); else alert('No coupon buttons found');
  • Citi Offers
    Clicks the plus icon to add all available offers. It uses a random delay between clicks to mimic human interaction.
    • To adjust delay speed, change: Math.random()*500+200
    Code:
    javascript:btns=[...document.querySelectorAll('svg path[d^="M12 4.2a"]')].map(p=>p.closest('button, [role="button"]')).filter(b=>b&&b.offsetParent!==null);if(!btns.length){alert('No Citi offer buttons found.');}else{let count=btns.length;c=()=>{b=btns.shift();if(!b){alert('Finished adding '+count+' Citi offers.');return;}b.click();setTimeout(c,Math.random()*500+200)};c();}
  • Chase Offers Auto-Adder Bookmarklet

    This bookmarklet automatically adds all available Chase credit card offers to your account.

    How to use:
    1. Create a new bookmark in your browser
    2. Name it "Chase Auto Offers" or something similar
    3. Instead of a URL, copy and paste the ENTIRE code from the Pastebin link below into the URL/location field
    4. Go to the Chase website and log in
    5. Navigate to your offers page
    6. Click the bookmarklet to run it

    Pastebin Link (Recommended Method):
    https://pastebin.com/raw/5j5CTFUg

    Alternative Method - Code (Not Recommended):
    Code:
    javascript:void(async function(){try{const c={ALL_OFFERS_URL:"https://secure.chase.com/web/auth/dashboard#/dashboard/merchantOffers/offerCategoriesPage?offerCategoryName=ALL",readinessTimeout:18e3,readinessPoll:400,detailObserveTotal:9e3,detailObservePoll:400,inlineTimeout:5200,poll:420,betweenOffers:450,detailBackDelay:1300,postReturnSettle:900,maxPerTileAttempts:3,maxPasses:8,pauseAfterReturn:800},p=new Set(["M24.07 4.34c-11.046 0-20 8.954-20 20s8.954 20 20 20 20-8.954 20-20c-.011-11.041-8.959-19.989-20-20m0 37c-9.389 0-17-7.611-17-17s7.611-17 17-17 17 7.611 17 17c-.013 9.383-7.617 16.987-17 17","M33.62 24.04a1.5 1.5 0 0 1-1.5 1.5h-6.53l-.02 6.52a1.5 1.5 0 0 1-3-.01l.02-6.51h-6.47a1.5 1.5 0 1 1 0-3h6.48l.02-6.51a1.5 1.5 0 0 1 1.5-1.49h.01a1.5 1.5 0 0 1 1.49 1.5l-.02 6.5h6.52a1.5 1.5 0 0 1 1.5 1.5"]),a=new Map;let n=1;function s(e,t=4e3){let o=document.getElementById("chase-offer-notifier");o||(o=document.createElement("div"),o.id="chase-offer-notifier",Object.assign(o.style,{position:"fixed",top:"20px",right:"20px",backgroundColor:"#0070ba",color:"white",padding:"15px 25px",borderRadius:"8px",boxShadow:"0 4px 12px rgba(0,0,0,0.2)",zIndex:"99999",fontSize:"16px",fontFamily:"sans-serif",opacity:"0",transition:"opacity 0.5s ease-in-out, top 0.5s ease-in-out"}),document.body.appendChild(o)),o.textContent=e,setTimeout(()=>{o.style.top="20px",o.style.opacity="1"},50),t>0&&setTimeout(()=>{o.style.top="0px",o.style.opacity="0"},t)}const d=()=>Array.from(document.querySelectorAll('div[data-testid="commerce-tile"]')),f=()=>/offerCategoriesPage.*offerCategoryName=ALL/i.test(location.href),h=()=>/offer-activated|offer-detail/i.test(location.href),u=()=>/offer-activated/i.test(location.href);function i(e){return e.dataset.offerAutoId||(e.dataset.offerAutoId="o_"+n++),e.dataset.offerAutoId}function l(e){return Array.from(e.querySelectorAll("path")).some(e=>p.has(e.getAttribute("d")))}function g(e){return Array.from(e.querySelectorAll('button,[role="button"]')).some(e=>{const t=(e.innerText||"")+" "+(e.getAttribute("aria-label")||"");return/add offer|activate|add to card/i.test(t.toLowerCase())})}function m(e){if(!e)return"";let t=e.match(/^\s*\d+\s+of\s+\d+\s+(.+?)\s+(?:\d+%|\$\d+)/);if(t)return t[1].trim();if(t=e.match(/^\s*\d+\s+of\s+\d+\s+(.+?)\s+Add offer/i))return t[1].trim();if(t=e.match(/^\s*\d+\s+of\s+\d+\s+(.+?)\s+(?:\d+%|\$\d+)\s+cash back/i))return t[1].trim();return e.split(/Add offer|Added|Success Added/i)[0].replace(/^\s*\d+\s+of\s+\d+\s+/,"").trim()||""}function b(e){const t=e.getAttribute("aria-label")||"",o=m(t),r=/(Added|Success Added)\s*$/i.test(t.trim()),a=/add offer/i.test(t)||l(e)||g(e);return{merchant:o,added:r,addableSignal:a}}async function w(){f()||(location.href=c.ALL_OFFERS_URL,await new Promise(e=>setTimeout(e,1100))),f()||(location.assign(c.ALL_OFFERS_URL),await new Promise(e=>setTimeout(e,1500)))}async function T(e=1){const t=Date.now();for(;Date.now()-t=e)return!0;await new Promise(e=>setTimeout(e,c.readinessPoll))}return!1}async function A(e){if(!e)return;e.scrollIntoView({block:"center"}),await new Promise(e=>setTimeout(e,200));try{e.focus({preventScroll:!0})}catch(e){}for(const t of["mouseover","mousedown","mouseup","click"])e.dispatchEvent(new MouseEvent(t,{bubbles:!0,cancelable:!0,view:window}))}function O(){return!!h()||(!!document.querySelector('#back-button, .navigation-bar__back-button, button[aria-label*="Back" i], button[aria-label*="Close" i]')||Array.from(document.querySelectorAll('button,[role="button"]')).some(e=>{const t=(e.innerText||"")+" "+(e.getAttribute("aria-label")||"");return/add offer|activate|add to card/i.test(t.toLowerCase())&&!e.closest('div[data-testid="commerce-tile"]')}))}function y(){return Array.from(document.querySelectorAll('button,[role="button"]')).find(e=>{const t=(e.innerText||"")+" "+(e.getAttribute("aria-label")||"");return/add offer|activate|add to card/i.test(t.toLowerCase())&&!/added/i.test(t.toLowerCase())&&!e.disabled})}function S(){return!!u()||(!!Array.from(document.querySelectorAll('[aria-live], [role="status"]')).some(e=>/added/i.test(e.textContent||"")&&!/add offer/i.test(e.textContent||""))||Array.from(document.querySelectorAll('button,[role="button"]')).some(e=>{const t=(e.innerText||"")+" "+(e.getAttribute("aria-label")||"");return/added/i.test(t)&&!/add offer/i.test(t)}))}async function v(){const e=Date.now();for(;Date.now()-esetTimeout(e,c.detailObservePoll))}return S()}async function k(){if(S())return{status:"already"};for(let e=1;e<=3;e++){const e=y();if(e){if(await A(e),await v())return{status:"added"}}else if(S())return{status:"added"};await new Promise(e=>setTimeout(e,500))}return S()?{status:"added"}:{status:"unconfirmed"}}async function E(){const e=document.querySelector('#back-button, .navigation-bar__back-button, button[aria-label*="Back" i], button[aria-label*="Close" i]');e?(await A(e),await new Promise(e=>setTimeout(e,700)),O()&&history.back()):history.back(),await new Promise(e=>setTimeout(e,c.pauseAfterReturn)),await T(1),await new Promise(e=>setTimeout(e,c.postReturnSettle))}function C(e){return a.get(i(e))||0}function R(e){const t=i(e);a.set(t,C(e)+1)}function L(e){return e.querySelector('button,[role="button"],a,div[onclick],span[onclick],div[tabindex]')||e}async function P(e){const t=Date.now();for(;Date.now()-t<3e3;){if(d().some(t=>{const o=b(t);return o.merchant===e&&o.added}))return!0;await new Promise(e=>setTimeout(e,250))}return!1}async function j(e,t){e.dispatchEvent(new MouseEvent("mouseenter",{bubbles:!0})),await new Promise(e=>setTimeout(e,90));const o=b(e);if(o.added)return{status:"already"};if(!o.addableSignal&&!t)return{status:"skip"};if(C(e)>=c.maxPerTileAttempts)return{status:"exhausted"};R(e),await A(L(e));const r=Date.now();for(;Date.now()-rsetTimeout(e,c.poll)),O())break;if(!e.isConnected)return{status:"ambiguous"};if(b(e).added)return{status:"added"}}if(O()){await new Promise(e=>setTimeout(e,c.detailBackDelay));const t=await k();return"added"===t.status||"already"===t.status?(await E(),await P(o.merchant),{status:"added"}):(await E(),{status:"unconfirmed"})}return{status:"ambiguous"}}async function M(){let e=0,t=!1;for(let o=1;o<=c.maxPasses;o++){let r=!1,a=new Set,n=!0;for(;n;){if(!d().length&&await T(1),!(h=d().find(e=>!a.has(i(e))))){n=!1;break}a.add(i(h)),b(h).added||(u=await j(h,t),"added"===u.status&&(e++,r=!0),await new Promise(e=>setTimeout(e,c.betweenOffers)))}var h,u;const l=d().filter(e=>!b(e).added).length;if(0===l)return void s(`Complete! Added ${e} offers.`,5e3);if(!r){if(t)return void s(`Stalled. Added ${e}. Remaining: ${l}.`,6e3);t=!0}}s(`Finished. Added ${e}. Remaining: ${d().filter(e=>!b(e).added).length}.`,6e3)}if(!/secure\.chase\.com/i.test(location.hostname))return void s("Error: Must be on a Chase.com page.",5e3);s("Starting Chase Offers Auto-Adder...",3e3),await w(),await T(1)?(d().forEach(i),await M()):s("Error: Could not find any offers on the page.",5e3)}catch(e){console.error("Chase Offers Script Failed:",e),s(`Script Error: ${e.message}. See console.`,8e3)}})();
    ⚠️ IMPORTANT NOTE ⚠️
    Special characters in the code above will be converted by the forum software, making the bookmarklet NOT work if copied directly from here. You have two options:

    1. Recommended: Use the Pastebin link above to get the full, working code
    2. Alternative: Edit this wiki post, click "Edit" and then copy the code from inside the code tags in the editor (NOT from the rendered view)

    Details:
    This bookmarklet navigates to your Chase offers page and automatically adds all available offers to your account. It includes intelligent retries and progress notifications.
  • More Coming Soon
    Add additional store names below to request new bookmarklets.

Your comment cannot be blank.

Sign up for a Slickdeals account to remove this ad.

Joined Jun 2005
L9: Master
> bubble2 4,541 Posts
12,362 Reputation
Original Poster
Pro
Expert
This user is an Expert in Home & Home Improvement
Cobalt_Blue_FF
09-04-2025 at 02:57 PM.
09-04-2025 at 02:57 PM.
Quote from bleuiko :
Do you have anything for Chase offers?
I was able to get chase to work, and added it to the wiki. Note the slightly different instructions. Let me know if you have any issues!
Reply
Joined Dec 2004
L7: Teacher
> bubble2 2,058 Posts
414 Reputation
bleuiko
09-04-2025 at 04:21 PM.
09-04-2025 at 04:21 PM.
Quote from Cobalt_Blue_FF :
I was able to get chase to work, and added it to the wiki. Note the slightly different instructions. Let me know if you have any issues!
Dude, legend! At work but trying it out tonight!
Reply
Joined Nov 2025
New User
> bubble2 2 Posts
10 Reputation
GreyPanther128
11-23-2025 at 08:21 PM.
11-23-2025 at 08:21 PM.
This is amazing, thanks! Is there any way to have the Amex Offers Clipper add offers for multiple cards at once? I know these scripts break every few months when Amex updates the page.
Reply
Joined Nov 2025
New User
> bubble2 2 Posts
10 Reputation
GreyPanther128
11-23-2025 at 08:26 PM.
11-23-2025 at 08:26 PM.
These are amazing, thank you! Is there any way to have the Amex Offers Clipper add offers for multiple cards at once? I know these scripts break every few months when Amex updates the page.
Reply
Joined Feb 2013
L7: Teacher
> bubble2 2,605 Posts
2,440 Reputation
Pro
Frugie
11-24-2025 at 03:52 AM.
11-24-2025 at 03:52 AM.
Coupon clipping automation for Shoprite and BJs Wholesale coupons would be great!
Reply
Page 4 of 4
Start the Conversation
 

More Grocery Deals & Discounts

Link Copied

The link has been copied to the clipboard.