This script has been deprecated! ⚠️

To add liking and saving to your site, we've released a new script which is better & more reliable! To check it out, head to Memberscript #106

Go To Memberscript #106

#20 - Save & Unsave CMS Items v0.3

Allow your members to like/unlike CMS items and save to their JSON!

Ver demostración

Optimized For Dynamic Content

This is usually what you will want to use - it works with over 100 CMS Items using Finsweet CMS Load.


<!-- 💙 MEMBERSCRIPT #20 v0.3 💙 SAVE & UNSAVE CMS ITEMS (OPTIMIZED FOR DYNAMIC CONTENT SUPPORT) -->
<script>
document.addEventListener("DOMContentLoaded", async function() {
  const memberstack = window.$memberstackDom;
  const memberData = (await memberstack.getMemberJSON()).data || {};

  const updateButtonVisibility = function(button, savedItems) {
    const itemId = button.getAttribute('ms-code-save-child');
    const isItemSaved = savedItems.includes(itemId);
    button.style.display = isItemSaved ? 'none' : 'block';
    const action = isItemSaved ? 'block' : 'none';
    button.closest('[ms-code-save]').querySelectorAll(`[ms-code-unsave-child="${itemId}"]`).forEach(unsaveButton => {
      unsaveButton.style.display = action;
    });
  };

  const toggleLikeButton = async function(button, jsonGroup, savedItems) {
    const itemId = button.getAttribute('ms-code-save-child');
    const itemIndex = savedItems.indexOf(itemId);

    if (itemIndex > -1) {
      savedItems.splice(itemIndex, 1); // Remove item from saved list
    } else {
      savedItems.push(itemId); // Add item to saved list
    }

    memberData[jsonGroup] = savedItems; // Directly modify the memberData to ensure updates are propagated correctly

    await memberstack.updateMemberJSON({ json: memberData });
    updateButtonVisibility(button, savedItems);
  };

  const initializeButtons = function(buttons, action) {
    buttons.forEach(button => {
      const jsonGroup = button.getAttribute('ms-code-save') || button.closest('[ms-code-save]').getAttribute('ms-code-save');
      const savedItems = memberData[jsonGroup] || [];
      if (action === 'update') updateButtonVisibility(button, savedItems);
      button.removeEventListener('click', buttonClickHandler); // Prevent multiple bindings
      button.addEventListener('click', buttonClickHandler);
    });
  };

  function buttonClickHandler(event) {
    event.preventDefault();
    const button = event.currentTarget;
    const jsonGroup = button.getAttribute('ms-code-save') || button.closest('[ms-code-save]').getAttribute('ms-code-save');
    const savedItems = memberData[jsonGroup] || [];
    toggleLikeButton(button, jsonGroup, savedItems);
  }

  const observeDOMChanges = function() {
    const observer = new MutationObserver((mutations) => {
      mutations.forEach((mutation) => {
        mutation.addedNodes.forEach(node => {
          if (node.nodeType === 1 && (node.matches('[ms-code-save-child]') || node.querySelector('[ms-code-save-child]'))) {
            const saveButtons = node.querySelectorAll('[ms-code-save-child]');
            const unsaveButtons = node.querySelectorAll('[ms-code-unsave-child]');
            initializeButtons(saveButtons, 'update');
            initializeButtons(unsaveButtons, 'attach');
          }
        });
      });
    });

    observer.observe(document.body, {
      childList: true,
      subtree: true
    });
  };

  const saveButtons = document.querySelectorAll('[ms-code-save-child]');
  const unsaveButtons = document.querySelectorAll('[ms-code-unsave-child]');

  initializeButtons(saveButtons, 'update');
  initializeButtons(unsaveButtons, 'attach');
  observeDOMChanges(); // Start observing changes in the document
});
</script>

Using Finsweet CMS Load

If you do have over 100 CMS items, please watch this video.

Loads after member object

If both logged in & logged out visitors will be on the page with this script, using this method will prevent errors.


<!-- 💙 MEMBERSCRIPT #20.1 v0.2 💙 SAVE & UNSAVE CMS ITEMS AFTER MEMBERSTACK LOAD -->
<script>
const memberstack = window.$memberstackDom;

const updateButtonVisibility = async function(button, jsonGroup) {
  const itemId = button.getAttribute('ms-code-save-child');
  const member = await memberstack.getMemberJSON();

  const savedItems = member.data && member.data[jsonGroup] ? member.data[jsonGroup] : [];
  const isItemSaved = savedItems.includes(itemId);

  const saveButton = button;
  const parentElement = button.closest('[ms-code-save]');
  const unsaveButtons = parentElement.querySelectorAll(`[ms-code-unsave-child="${itemId}"]`);

  unsaveButtons.forEach(unsaveButton => {
    if (isItemSaved) {
      saveButton.style.display = 'none';
      unsaveButton.style.display = 'block';
    } else {
      saveButton.style.display = 'block';
      unsaveButton.style.display = 'none';
    }
  });
};

const toggleLikeButton = async function(button, jsonGroup) {
  const itemId = button.getAttribute('ms-code-save-child');
  const member = await memberstack.getMemberJSON();

  if (!member.data) {
    member.data = {};
  }

  if (!member.data[jsonGroup]) {
    member.data[jsonGroup] = [];
  }

  const isItemSaved = member.data[jsonGroup].includes(itemId);

  const parentElement = button.closest('[ms-code-save]');
  const unsaveButtons = parentElement.querySelectorAll(`[ms-code-unsave-child="${itemId}"]`);

  if (isItemSaved) {
    member.data[jsonGroup] = member.data[jsonGroup].filter(item => item !== itemId);
    button.style.display = 'block';
    unsaveButtons.forEach(unsaveButton => {
      unsaveButton.style.display = 'none';
    });
  } else {
    member.data[jsonGroup].push(itemId);
    button.style.display = 'none';
    unsaveButtons.forEach(unsaveButton => {
      unsaveButton.style.display = 'block';
    });
  }

  await memberstack.updateMemberJSON({
    json: member.data
  });

  updateButtonVisibility(button, jsonGroup);
};

memberstack.getCurrentMember().then(({ data }) => {
  if (data) {
    // Member is logged in
    const saveButtons = document.querySelectorAll('[ms-code-save-child]');

    saveButtons.forEach(button => {
      const jsonGroup = button.getAttribute('ms-code-save') || button.closest('[ms-code-save]').getAttribute('ms-code-save');
      updateButtonVisibility(button, jsonGroup);
      button.addEventListener('click', async function(event) {
        event.preventDefault();
        await toggleLikeButton(button, jsonGroup);
      });
    });

    const unsaveButtons = document.querySelectorAll('[ms-code-unsave-child]');

    unsaveButtons.forEach(button => {
      const jsonGroup = button.getAttribute('ms-code-save') || button.closest('[ms-code-save]').getAttribute('ms-code-save');
      button.addEventListener('click', async function(event) {
        event.preventDefault();
        const parentElement = button.closest('[ms-code-save]');
        const saveButton = parentElement.querySelector(`[ms-code-save-child="${button.getAttribute('ms-code-unsave-child')}"]`);
        await toggleLikeButton(saveButton, jsonGroup);
      });
    });
  } else {
    // If member is not logged in
  }
});
</script>

Creación del escenario Make.com

1. Descargue el proyecto JSON a continuación para empezar.

2. Navegue hasta Make.com y Cree un nuevo escenario...

3. Haga clic en el pequeño cuadro con 3 puntos y luego Importar Blueprint...

4. Sube tu archivo y ¡voilá! Ya está listo para vincular sus propias cuentas.

¿Necesitas ayuda con este MemberScript?

Todos los clientes de Memberstack pueden solicitar asistencia en el Slack 2.0. Tenga en cuenta que no se trata de funciones oficiales y que no se puede garantizar la asistencia.

Únete al Slack 2.0
Notas de la versión

#20.1 - New version released

This new version is another option which will load the script AFTER the member object has loaded. This is a better script for pages which both visitors and members will be viewing.

v0.2 - Fixed strange behavior

Having multiple of the same list on a page caused an issue with button visibility.

v0.3 - Performance improvements + dynamic content support

Changes have been made which drastically improves the loading speed, along with allowing dynamic content from pagination.

Atributos
Descripción
Atributo
No se han encontrado artículos.
Guías / Tutoriales
No se han encontrado artículos.
Tutorial
¿Qué es Memberstack?

Autenticación y pagos para sitios Webflow

Añada inicios de sesión, suscripciones, contenido cerrado y mucho más a su sitio Webflow: fácil y totalmente personalizable.

Más información

"We've been using Memberstack for a long time, and it has helped us achieve things we would have never thought possible using Webflow. It's allowed us to build platforms with great depth and functionality and the team behind it has always been super helpful and receptive to feedback"

Jamie Debnam
39 Digital

"Been building a membership site with Memberstack and Jetboost for a client. Feels like magic building with these tools. As someone who’s worked in an agency where some of these apps were coded from scratch, I finally get the hype now. This is a lot faster and a lot cheaper."

Félix Meens
Webflix Studio

"One of the best products to start a membership site - I like the ease of use of Memberstack. I was able to my membership site up and running within a day. Doesn't get easier than that. Also provides the functionality I need to make the user experience more custom."

Eric McQuesten
Health Tech Nerds
Off World Depot

"My business wouldn't be what it is without Memberstack. If you think $30/month is expensive, try hiring a developer to integrate custom recommendations into your site for that price. Incredibly flexible set of tools for those willing to put in some minimal efforts to watch their well put together documentation."

Riley Brown
Off World Depot

"The Slack community is one of the most active I've seen and fellow customers are willing to jump in to answer questions and offer solutions. I've done in-depth evaluations of alternative tools and we always come back to Memberstack - save yourself the time and give it a shot."

Abadía Burtis
Health Tech Nerds
Slack

Need help with this MemberScript? Join our Slack community!

Únete al Slack de la comunidad Memberstack y ¡pregunta! Espera una respuesta rápida de un miembro del equipo, un experto de Memberstack o un compañero de la comunidad.

Únete a nuestro Slack