Skip to Content
Odoo Menu
  • Zaloguj się
  • Wypróbuj za darmo
  • Aplikacje
    Finanse
    • Księgowość
    • Fakturowanie
    • Wydatki
    • Arkusz kalkulacyjny (BI)
    • Dokumenty
    • Podpisy
    Sprzedaż
    • CRM
    • Sprzedaż
    • PoS Sklep
    • PoS Restauracja
    • Subskrypcje
    • Wypożyczalnia
    Strony Internetowe
    • Kreator Stron Internetowych
    • eCommerce
    • Blog
    • Forum
    • Czat na Żywo
    • eLearning
    Łańcuch dostaw
    • Magazyn
    • Produkcja
    • PLM
    • Zakupy
    • Konserwacja
    • Jakość
    Zasoby Ludzkie
    • Pracownicy
    • Rekrutacja
    • Urlopy
    • Ocena pracy
    • Polecenia Pracownicze
    • Flota
    Marketing
    • Marketing Społecznościowy
    • E-mail Marketing
    • SMS Marketing
    • Wydarzenia
    • Automatyzacja Marketingu
    • Ankiety
    Usługi
    • Projekt
    • Ewidencja czasu pracy
    • Usługi Terenowe
    • Helpdesk
    • Planowanie
    • Spotkania
    Produktywność
    • Dyskusje
    • Zatwierdzenia
    • IoT
    • VoIP
    • Baza wiedzy
    • WhatsApp
    Aplikacje trzecich stron Studio Odoo Odoo Cloud Platform
  • Branże
    Sprzedaż detaliczna
    • Księgarnia
    • Sklep odzieżowy
    • Sklep meblowy
    • Sklep spożywczy
    • Sklep z narzędziami
    • Sklep z zabawkami
    Żywienie i hotelarstwo
    • Bar i Pub
    • Restauracja
    • Fast Food
    • Pensjonat
    • Dystrybutor napojów
    • Hotel
    Agencja nieruchomości
    • Agencja nieruchomości
    • Biuro architektoniczne
    • Budowa
    • Zarządzanie nieruchomościami
    • Ogrodnictwo
    • Stowarzyszenie właścicieli nieruchomości
    Doradztwo
    • Biuro księgowe
    • Partner Odoo
    • Agencja marketingowa
    • Kancelaria prawna
    • Agencja rekrutacyjna
    • Audyt i certyfikacja
    Produkcja
    • Tekstylia
    • Metal
    • Meble
    • Jedzenie
    • Browar
    • Prezenty firmowe
    Zdrowie & Fitness
    • Klub sportowy
    • Salon optyczny
    • Centrum fitness
    • Praktycy Wellness
    • Apteka
    • Salon fryzjerski
    Transakcje
    • Złota rączka
    • Wsparcie Sprzętu IT
    • Systemy energii słonecznej
    • Szewc
    • Firma sprzątająca
    • Usługi HVAC
    Inne
    • Organizacja non-profit
    • Agencja Środowiskowa
    • Wynajem billboardów
    • Fotografia
    • Leasing rowerów
    • Sprzedawca oprogramowania
    Przeglądaj wszystkie branże
  • Community
    Ucz się
    • Samouczki
    • Dokumentacja
    • Certyfikacje
    • Szkolenie
    • Blog
    • Podcast
    Pomóż w nauce innym
    • Program Edukacyjny
    • Scale Up! Gra biznesowa
    • Odwiedź Odoo
    Skorzystaj z oprogramowania
    • Pobierz
    • Porównaj edycje
    • Wydania
    Współpracuj
    • Github
    • Forum
    • Wydarzenia
    • Tłumaczenia
    • Zostań partnerem
    • Usługi dla partnerów
    • Zarejestruj swoją firmę rachunkową
    Skorzystaj z usług
    • Znajdź partnera
    • Znajdź księgowego
    • Spotkaj się z doradcą
    • Usługi wdrożenia
    • Opinie klientów
    • Wsparcie
    • Aktualizacje
    Github Youtube Twitter Linkedin Instagram Facebook Spotify
    +1 (650) 691-3277
    Zaplanuj demo
  • Cennik
  • Pomoc

Odoo is the world's easiest all-in-one management software.
It includes hundreds of business apps:

  • CRM
  • e-Commerce
  • Księgowość
  • Zapasy
  • PoS
  • Projekt
  • MRP
All apps
Musisz się zarejestrować, aby móc wchodzić w interakcje z tą społecznością.
Wszystkie posty Osoby Odznaki
Tagi (Zobacz wszystko)
odoo accounting v14 pos v15
O tym forum
Musisz się zarejestrować, aby móc wchodzić w interakcje z tą społecznością.
Wszystkie posty Osoby Odznaki
Tagi (Zobacz wszystko)
odoo accounting v14 pos v15
O tym forum
Pomoc

Custom Cards Payment Module: POS Payment Summary Not Updating with Installments

Zaprenumeruj

Otrzymaj powiadomienie o aktywności w tym poście

To pytanie dostało ostrzeżenie
javascriptdevelopmentsalesproject
2 Odpowiedzi
2187 Widoki
Awatar
Diego Naranjo

Hey there! I'm having a tricky issue with my custom Cards payment module for Odoo 18 POS. I built this module to handle credit card payments with installments, and everything works great except for one annoying detail in the payment screen. The total amount in the payment summary doesn't update automatically when selecting installments. This is my custom implementation that extends the default POS payment screen, using a CardsPaymentHandler class to manage card brands and installment calculations. The weird part is that the amount only updates in two scenarios:

If I remove the payment method and add it again (then it shows the correct total from the order/status)

If I manually type numbers using the POS numeric keypad (then it updates the payment method total in the summary)

All the interest calculations work fine, and I can see the correct amounts in the state, but for some reason, the payment summary display won't refresh with the new total when choosing installments.

Has anyone encountered something similar when extending the POS payment screen or knows what might be missing? Maybe there's a specific method I need to call to force the payment summary update?

I published the main functions that manage the updates of the totals according to the installments.

Thanks in advance!

/**

    * Handles card brand selection change

    * Resets prices and loads available installments

    * @param {Event} ev Change event from select element

    */

async onCardBrandChange(ev) {

    const cardBrand = ev.target.value;

    this._resetPrices();

    if (!cardBrand) {

        this._resetState();

        return;

    }


    try {

        const amount = this.currentOrder.get_total_with_tax();

        const installments = await rpc('/pos_payment_method/get_installments', {

            payment_method_id: CARDS_PAYMENT_METHOD_ID,

            card_brand: cardBrand,

            amount: amount

        });


        if (!installments) {

            throw new Error('There isn´t installments plan.');

        }


        this.state.selectedCardBrand = cardBrand;

        this.state.installmentOptions = this._formatInstallmentOptions(installments);

        this.screen.render();

    } catch (error) {

        this._resetState();

        this.showError('Error loading installments');

    }

}


/**

    * Handles installment selection change

    * Updates order amounts and payment line with interest

    * @param {Event} ev Change event from select element

    */

onInstallmentChange(ev) {

    const installments = ev.target.value;

    const option = this.state.installmentOptions.find(

        opt => opt.installments.toString() === installments

    );


    if (!option) return;


    try {

        // Update state with selected installment data

        Object.assign(this.state, {

            selectedInstallments: option.installments,

            installmentAmount: parseFloat(option.amount),

            totalWithInterest: parseFloat(option.total),

            interestRate: parseFloat(option.rate)

        });

        if (typeof option.coefficient === 'number' && !isNaN(option.coefficient)) {

            this._updateOrderLinesWithInterest(option.coefficient);

            const totalWithInterest = parseFloat(this.state.totalWithInterest);

            if (!isNaN(totalWithInterest) && this.selectedPaymentLine) {

                this.selectedPaymentLine.amount = totalWithInterest;

                this.selectedPaymentLine.set_payment_status("done");

                if (this.screen.numberBuffer) {

                    this.screen.numberBuffer.set(totalWithInterest.toString());

                }

            }

        }

        this._updatePaymentSummary();

        this.screen.render();

    } catch (error) {

        this.showError('Error processing installment change');

    }

}


/**

    * Updates order lines with interest coefficient

    * Recalculates totals and updates UI

    * @param {number} coefficient Interest multiplier

    */

_updateOrderLinesWithInterest(coefficient) {

    const order = this.currentOrder;

    if (!order) return;


    const coef = parseFloat(coefficient);

    if (isNaN(coef)) return;


    order.get_orderlines().forEach(line => {

        try {

            if (!line.original_price) {

                line.original_price = line.get_unit_price();

            }

            line.set_unit_price(line.original_price * coef);

        } catch (error) {

            console.error('Error processing line:', error);

        }

    });


    order.recomputeOrderData();

    this.state.totalWithInterest = order.amount_total;

    this._forcePaymentSummaryUpdate();

}


/**

    * Forces payment summary update in UI

    * Updates totals, remaining amount and payment lines

    */

_forcePaymentSummaryUpdate() {

    if (!this.currentOrder) return;


    try {

        Object.assign(this.screen.state, {

            totalAmount: this.currentOrder.amount_total,

            remainingAmount: this.currentOrder.get_due(),

            paymentLines: [...this.currentOrder.payment_ids]

        });

        if (this.screen.numberBuffer) {

            this.screen.numberBuffer.set(this.state.totalWithInterest.toString());

        }

        this.screen.render();

    } catch (error) {

        this.showError('Error updating payment summary');

    }

}


/**

    * Updates payment summary with new totals

    * Handles payment line amount updates

    */

_updatePaymentSummary() {

    if (!this.currentOrder || !this.selectedPaymentLine) return;

    const paid = this.currentOrder.get_total_paid();

    const remaining = Math.max(this.state.totalWithInterest - paid, 0);

    Object.assign(this.screen.state, {

        totalAmount: this.state.totalWithInterest,

        remainingAmount: remaining,

        paymentLines: [...this.currentOrder.payment_ids]

    });

    this.selectedPaymentLine.set_amount(this.state.totalWithInterest);

    this.currentOrder.set_total_with_tax(this.state.totalWithInterest);

    this.screen.payment_interface.update_payment_summary({

        total: this.state.totalWithInterest,

        total_paid: paid,

        remaining: remaining

    });

    this.screen.render();

}


/**

    * Updates payment line amount

    * Handles amount validation and UI updates

    * @param {number|false} amount New amount or false to get from buffer

    */

updatePaymentLine(amount = false) {

    this.updateReferences();

    if (amount === false) {

        amount = this.screen.numberBuffer.getFloat();

    }

    if (amount === null) {

        this._resetPrices();

        this.screen.deletePaymentLine(this.screen.selectedPaymentLine.uuid);

        return;

    }

    const maxAmount = this.state.totalWithInterest || this.currentOrder.get_total_with_tax();

    if (amount > maxAmount) {

        amount = maxAmount;

        this.screen.numberBuffer.set(maxAmount.toString());

        this._showMaxAmountError();

    }

    this.screen.selectedPaymentLine.set_amount(amount);

    this.screen.render();

}


/**

    * Resets prices to original values

    * Clears interest calculations

    */

_resetPrices() {

    const order = this.currentOrder;

    if (!order) return;

    order.get_orderlines().forEach(line => {

        try {

            if (line.original_price) {

                line.set_unit_price(line.original_price);

                line.setLinePrice();

            }

        } catch (error) {

            console.error('Error resetting price:', error);

        }

    });

    order.recomputeOrderData();

}

0
Awatar
Odrzuć
Diego Naranjo
Autor

Hi there!, after 4 months i managed to make an absolutely FREE module for use with credit cards through the Fiserv gateway service (at the moment it only works for Argentina but it could be improved for other countries).

It is published on github, you can find it as fiserv_gateway.

The ideal is that we can all improve it as a community so that it works in the countries that see it useful.
It still has several points to improve, but it is a good starting point for all those who need a form of payment online and at the POS with debit/credit cards.
Any queries can locate my contact details on my github profile.
Kind regards

Awatar
Diego Naranjo
Autor Najlepsza odpowiedź

Hello Tiago,

I thank you for your kindness, but I have already tried all the options provided by the available AI and I have not been able to make progress.

Kind regards.

0
Awatar
Odrzuć
Awatar
Tiago de Freitas Gomes
Najlepsza odpowiedź

The analysis of your code indicates that the main functions to update the payment summary are in place, but there might be issues related to the DOM rendering cycle or state synchronization in the Odoo POS. Here are the potential causes and solutions for the issue of the payment summary not refreshing when selecting installments:


Potential Causes

  1. DOM Rendering Cycle Issue
    • The DOM might not be updating correctly after the installment calculations. This happens when state changes are not propagated to the user interface components.
  2. Update Methods Not Called Properly
    • Although _updatePaymentSummary is being called, it might require a more explicit method to ensure the POS updates the payment component.
  3. numberBuffer Synchronization
    • The numeric buffer synchronization with the new totals might not be functioning correctly, requiring a more explicit approach to reflect the updated amounts.

Fixes and Improvements

1. Force Screen Refresh

Add an explicit command to force the payment component to re-render after updates:

_updatePaymentSummary() {
    if (!this.currentOrder || !this.selectedPaymentLine) return;

    const paid = this.currentOrder.get_total_paid();
    const remaining = Math.max(this.state.totalWithInterest - paid, 0);

    Object.assign(this.screen.state, {
        totalAmount: this.state.totalWithInterest,
        remainingAmount: remaining,
        paymentLines: [...this.currentOrder.payment_ids],
    });

    this.selectedPaymentLine.set_amount(this.state.totalWithInterest);
    this.currentOrder.set_total_with_tax(this.state.totalWithInterest);

    // Explicitly update the payment interface
    if (this.screen.payment_interface) {
        this.screen.payment_interface.update_payment_summary({
            total: this.state.totalWithInterest,
            total_paid: paid,
            remaining: remaining,
        });
    }

    // Force a screen re-render
    this.screen.render(true);
}

2. Update numberBuffer

Ensure the numberBuffer is synchronized with the recalculated totals:

if (this.screen.numberBuffer) {
    this.screen.numberBuffer.reset();
    this.screen.numberBuffer.set(this.state.totalWithInterest.toString());
}

3. Explicitly Recalculate and Update Totals

After updating the installments, ensure the order data is fully recalculated:

this.currentOrder.recomputeOrderData();
this.state.totalWithInterest = this.currentOrder.amount_total;
this._forcePaymentSummaryUpdate();

4. Force Global State Updates

If the state changes are not propagating correctly, add a mechanism to trigger a global update:

this.env.pos.trigger('update-screen');

Final Checks

Ensure that:

  1. POS Dependencies: There are no broken dependencies or modifications in your POS module that might prevent updates.
  2. Order and Lines: Update methods like recomputeOrderData are being called at the right time.
  3. Custom Events: If needed, create custom events to trigger component updates.

Enable Detailed Logging

If the issue persists, enable detailed logging to pinpoint where the state changes are not propagating correctly. For example:

console.log('Current State:', this.state);
console.log('Order Total:', this.currentOrder.amount_total);
console.log('Payment Lines:', this.currentOrder.payment_ids);

This will help isolate the problem. Tell me if it helped you.... 

by chatGPT





Another answer (outra resposta):


Analisando o código fornecido, posso identificar o problema e sugerir uma correção. O principal motivo pelo qual o sumário de pagamento não está atualizando automaticamente é devido à forma como as atualizações de estado estão sendo manipuladas.


(By analyzing the code provided, I can identify the problem and suggest a fix. The main reason the payment summary is not updating automatically is due to the way state updates are being handled.)


class CardsPaymentHandler {
    async onCardBrandChange(ev) {
        const cardBrand = ev.target.value;
        this._resetPrices();
       
        if (!cardBrand) {
            this._resetState();
            return;
        }

        try {
            const amount = this.currentOrder.get_total_with_tax();
            const installments = await rpc('/pos_payment_method/get_installments', {
                payment_method_id: CARDS_PAYMENT_METHOD_ID,
                card_brand: cardBrand,
                amount: amount
            });

            if (!installments) {
                throw new Error('Não há plano de parcelamento disponível.');
            }

            this.state.selectedCardBrand = cardBrand;
            this.state.installmentOptions = this._formatInstallmentOptions(installments);
            this._triggerUpdate();  // Nova função para garantir atualização
        } catch (error) {
            this._resetState();
            this.showError('Erro ao carregar parcelas');
        }
    }

    onInstallmentChange(ev) {
        const installments = ev.target.value;
        const option = this.state.installmentOptions.find(
            opt => opt.installments.toString() === installments
        );

        if (!option) return;

        try {
            Object.assign(this.state, {
                selectedInstallments: option.installments,
                installmentAmount: parseFloat(option.amount),
                totalWithInterest: parseFloat(option.total),
                interestRate: parseFloat(option.rate)
            });

            if (typeof option.coefficient === 'number' && !isNaN(option.coefficient)) {
                this._updateOrderLinesWithInterest(option.coefficient);
                this._updatePaymentLine(this.state.totalWithInterest);  // Atualiza linha de pagamento
            }
           
            this._triggerUpdate();  // Força atualização completa
        } catch (error) {
            this.showError('Erro ao processar alteração de parcelas');
        }
    }

    _updatePaymentLine(amount) {
        if (this.selectedPaymentLine) {
            this.selectedPaymentLine.amount = amount;
            this.selectedPaymentLine.set_payment_status("done");
        }
    }

    _triggerUpdate() {
        // Nova função para garantir atualização completa
        this._updatePaymentSummary();
        this.screen.render();
       
        // Força atualização do componente pai
        if (this.screen.props.update) {
            this.screen.props.update();
        }
       
        // Atualiza o buffer numérico se necessário
        if (this.screen.numberBuffer) {
            this.screen.numberBuffer.set(this.state.totalWithInterest.toString());
        }
    }

    _updatePaymentSummary() {
        if (!this.currentOrder) return;

        const totalWithInterest = this.state.totalWithInterest || this.currentOrder.get_total_with_tax();
        const paid = this.currentOrder.get_total_paid();
        const remaining = Math.max(totalWithInterest - paid, 0);

        // Atualiza o estado do pedido
        this.currentOrder.set_total_with_tax(totalWithInterest);

        // Atualiza o estado da tela
        Object.assign(this.screen.state, {
            totalAmount: totalWithInterest,
            remainingAmount: remaining,
            paymentLines: [...this.currentOrder.payment_ids]
        });

        // Atualiza a interface de pagamento
        this.screen.payment_interface.update_payment_summary({
            total: totalWithInterest,
            total_paid: paid,
            remaining: remaining
        });
    }
}

export default CardsPaymentHandler;


As principais correções e melhorias implementadas são:


Criação do método _triggerUpdate() para centralizar todas as atualizações necessárias

Melhor gerenciamento do estado e atualizações da interface

Atualização consistente da linha de pagamento

Tratamento mais robusto dos valores com juros

Garantia de que o sumário de pagamento seja sempre atualizado


Principais mudanças:


Centralização das atualizações em um único método _triggerUpdate()

Melhor tratamento das atualizações do numberBuffer

Adição de verificações de segurança para evitar erros

Garantia de que o componente pai seja notificado das mudanças

Atualização consistente do estado do pedido e da interface


Para implementar esta correção, você precisará:


Substituir seu código atual pelo novo

Certificar-se de que a classe está sendo corretamente exportada

Verificar se todas as dependências estão disponíveis

Testar especialmente os cenários de alteração de parcelas


(The main corrections and improvements implemented are:


Creation of the _triggerUpdate() method to centralize all necessary updates

Better state management and interface updates

Consistent payline update

More robust treatment of interest-bearing amounts

Ensuring that the payment summary is always updated


Main changes:


Centralization of updates in a single _triggerUpdate() method

Better handling of numberBuffer updates

Adding security checks to prevent errors

Ensuring that the parent component is notified of changes

Consistent order and interface state update


To implement this fix, you will need:


Replace your current code with the new one

Make sure the class is being exported correctly

Check if all dependencies are available

Specially test installment change scenarios.)

By Claude.ai

0
Awatar
Odrzuć
Tiago de Freitas Gomes

EI Diego!

Estou buscando uma solução para oferecer parcelamentos no meu site.
Estou no Brasil e aqui não existe a possibilidade de um e-commerce não oferecer parcelamento. Descobri esta semana que o Odoo tem essa limitação e fiquei bem desapontado.

Tenho resolvido muitos problemas dentro do Odoo com o auxílio das iAs, mas como não estou programando, não consigo checar se essas soluções funcionariam.

Seguirei acompanhando o seu post.

Podoba Ci się ta dyskusja? Dołącz do niej!

Stwórz konto dzisiaj, aby cieszyć się ekskluzywnymi funkcjami i wchodzić w interakcje z naszą wspaniałą społecznością!

Zarejestruj się
Powiązane posty Odpowiedzi Widoki Czynność
OWL Component Not Rendering in Custom Layout (Odoo 18) — Likely XML Template Issue
javascript development project debug
Awatar
Awatar
1
lip 25
3514
How can I use my only web url domain using odoo?
javascript development
Awatar
Awatar
1
wrz 25
1272
Service worker `fetch` fails in `install` event when trying to cache assets
webclient javascript development sales chrome
Awatar
Awatar
2
wrz 25
976
Automatically create a task and timesheet lines from a confirmed quotation in Odoo 17? Rozwiązane
sales project
Awatar
Awatar
Awatar
2
paź 25
2585
How to Load Related Many2Many Data (pos.ticket.type) into the Frontend?
javascript development
Awatar
0
cze 25
1679
Społeczność
  • Samouczki
  • Dokumentacja
  • Forum
Open Source
  • Pobierz
  • Github
  • Runbot
  • Tłumaczenia
Usługi
  • Hosting Odoo.sh
  • Wsparcie
  • Aktualizacja
  • Indywidualne rozwiązania
  • Edukacja
  • Znajdź księgowego
  • Znajdź partnera
  • Zostań partnerem
O nas
  • Nasza firma
  • Zasoby marki
  • Skontaktuj się z nami
  • Oferty pracy
  • Wydarzenia
  • Podcast
  • Blog
  • Klienci
  • Informacje prawne • Prywatność
  • Bezpieczeństwo Odoo
الْعَرَبيّة Català 简体中文 繁體中文 (台灣) Čeština Dansk Nederlands English Suomi Français Deutsch हिंदी Bahasa Indonesia Italiano 日本語 한국어 (KR) Lietuvių kalba Język polski Português (BR) română русский язык Slovenský jazyk slovenščina Español (América Latina) Español ภาษาไทย Türkçe українська Tiếng Việt

Odoo to pakiet aplikacji biznesowych typu open source, które zaspokoją wszystkie potrzeby Twojej firmy: CRM, eCommerce, księgowość, inwentaryzacja, punkt sprzedaży, zarządzanie projektami itp.

Unikalną wartością Odoo jest to, że jest jednocześnie bardzo łatwe w użyciu i w pełni zintegrowane.

Website made with

Odoo Experience on YouTube

1. Use the live chat to ask your questions.
2. The operator answers within a few minutes.

Live support on Youtube
Watch now