Skip to content

LinusBolls/splid-js

Repository files navigation

splid.js

a feature-complete typescript client for the Splid (https://splid.app) API. Splid is a free mobile app for keeping track of expenses among friend groups. this package is not officially associated with Splid.

last updated Nov 7 2024

Install

install the package:

npm install splid-js

Examples

// basic usage
import { SplidClient } from 'splid-js';

async function main() {
  const splid = new SplidClient();

  const inviteCode = 'PWJ E2B P7A';

  const groupRes = await splid.group.getByInviteCode(inviteCode);

  const groupId = groupRes.result.objectId;

  const groupInfo = await splid.groupInfo.getOneByGroup(groupId);
  const members = await splid.person.getAllByGroup(groupId);
  const expensesAndPayments = await splid.entry.getAllByGroup(groupId);

  const balance = SplidClient.getBalance(
    members,
    expensesAndPayments,
    groupInfo
  );
  const suggestedPayments = SplidClient.getSuggestedPayments(balance);

  console.log(balance, suggestedPayments);
}
main();
// parsing the returned data
import { SplidClient } from 'splid-js';

const formatCurrency = (amount: number, currency: string) => {
  return amount.toLocaleString(undefined, {
    style: 'currency',
    currency,
    minimumFractionDigits: 2,
    maximumFractionDigits: 2,
  });
};

const getEntryDescription = (
  entry: SplidJs.Entry,
  members: SplidJs.Person[]
) => {
  const primaryPayer = members.find((i) => i.GlobalId === entry.primaryPayer)!;

  for (const item of entry.items) {
    const totalAmount = item.AM;
    // a map of a userId to their share (how much they profit from the expense). the shares are floats between 0 and 1 and their sum is exactly 1.
    const userIdToShareMap = item.P.P;

    const profiteers = Object.entries(userIdToShareMap).map(
      ([userId, share]) => {
        const user = members.find((i) => i.GlobalId === userId)!;

        const shareText = formatCurrency(
          totalAmount * share,
          entry.currencyCode
        );
        return user.name + ' (' + shareText + ')';
      }
    );
    const profiteersText = profiteers.join(', ');

    const totalText = formatCurrency(totalAmount, entry.currencyCode);

    if (entry.isPayment) {
      const description =
        primaryPayer.name + ' sent ' + totalText + ' to ' + profiteersText;

      return description;
    } else {
      const description =
        primaryPayer.name + ' payed ' + totalText + ' for ' + profiteersText;

      return description;
    }
  }
};

async function main() {
  const splid = new SplidClient();

  const inviteCode = 'PWJ E2B P7A';

  const groupRes = await splid.group.getByInviteCode(inviteCode);

  const groupId = groupRes.result.objectId;

  const members = await splid.person.getAllByGroup(groupId);
  const expensesAndPayments = await splid.entry.getAllByGroup(groupId);

  for (const entry of expensesAndPayments) {
    console.log(getEntryDescription(entry, members));
  }
}
main();
// updating group properties
const groupInfo = await splid.groupInfo.getOneByGroup(groupId);

groupInfo.name = 'Modified Group 🔥';

groupInfo.customCategories.push('Pharmaceuticals 💊');

// setting the currency exchange rates to the most recent values
groupInfo.currencyRates = await splid.getCurrencyRates();

groupInfo.defaultCurrencyCode = 'EUR';

await splid.groupInfo.set(groupInfo);
// updating group wallpaper (NodeJs)
import fs from 'fs';

const file = await fs.promises.readFile('image.png');

const uploadRes = await splid.file.upload(file);

groupInfo.wallpaperID = uploadRes.dataID;
// updating person properties
const members = await splid.person.getAllByGroup(groupId);

const linus = members.find((i) => i.name === 'Linus');

linus.name = 'Alex';
linus.initials = 'A';

await splid.person.set(linus);
// updating entry properties
const entries = await splid.entry.getAllByGroup(groupId);

const pizzaEntries = entries.filter((i) =>
  i.title.toLowerCase().includes('pizza')
);

for (const entry of pizzaEntries) {
  // setting the category of an entry
  entry.category = {
    type: 'custom',
    originalName: 'Italian Food 🍕',
  };
  // setting the date of an entry
  if (!entry.date) {
    entry.date = {
      __type: 'Date',
      iso: new Date().toISOString(),
    };
  } else {
    entry.date.iso = new Date().toISOString();
  }
}
await splid.entry.set(pizzaEntries);
// converting all foreign currency entries to the default currency of the group

const foreignCurrencyEntries = expensesAndPayments.filter(
  (i) => !i.isPayment && i.currencyCode !== groupInfo.defaultCurrencyCode
);

for (const entry of foreignCurrencyEntries) {
  const factor =
    groupInfo.currencyRates[entry.currencyCode] /
    groupInfo.currencyRates[groupInfo.defaultCurrencyCode];

  entry.currencyCode = groupInfo.defaultCurrencyCode;

  for (const item of entry.items) {
    item.AM = item.AM * factor;
  }
  for (const [payerId, amount] of Object.entries(entry.secondaryPayers ?? {})) {
    entry.secondaryPayers[payerId] = amount * factor;
  }
}
await splid.entry.set(foreignCurrencyEntries);
// deleting entries
const entries = await splid.entry.getAllByGroup(groupId);

const entry = entries[0];

entry.isDeleted = true;

await splid.entry.set(entry);
// creating a group
await splid.group.create('🎉 Ramber Zamber', ['Linus', 'Laurin', 'Oskar']);
// creating a basic expense
await splid.entry.expense.create(
  {
    groupId,
    payers: [linus.GlobalId],
    title: 'döner',
  },
  {
    amount: 10,
    // equivalent to equivalent to { [laurin.GlobalId]: 0.5, [oskar.GlobalId]: 0.5 }
    profiteers: [laurin.GlobalId, oskar.GlobalId],
  }
);
// creating an expense with multiple items
await splid.entry.expense.create(
  {
    groupId,
    payers: [linus.GlobalId],
    title: 'shopping spree 😌',
  },
  [
    {
      title: 'gucci belt',
      amount: 10,
      profiteers: [laurin.GlobalId],
    },
    {
      title: 'drippy hat',
      amount: 15,
      profiteers: [oskar.GlobalId],
    },
  ]
);
// creating an expense with multiple payers (both pay half)
await splid.entry.expense.create(
  {
    groupId,
    // equivalent to { [linus.GlobalId]: 5, [oskar.GlobalId]: 5 }
    payers: [linus.GlobalId, oskar.GlobalId],
    title: 'döner',
  },
  {
    amount: 10,
    profiteers: [linus.GlobalId, oskar.GlobalId],
  }
);
// creating an expense with unevenly split payers (oskar pays 3€)
await splid.entry.expense.create(
  {
    groupId,
    // equivalent to { [linus.GlobalId]: 7, [oskar.GlobalId]: 3 }
    payers: [linus.GlobalId, { id: oskar.GlobalId, amount: 3 }],
    title: 'shopping',
  },
  {
    amount: 10,
    profiteers: [laurin.GlobalId],
  }
);
// creating an expense with unevenly split profiteers (oskar owes 2.25€)
await splid.entry.expense.create(
  {
    groupId,
    payers: [linus.GlobalId],
    title: 'shopping',
  },
  {
    amount: 10,
    // equivalent to { [linus.GlobalId]: 3 / 4, [oskar.GlobalId]: 1 / 4 }
    profiteers: [linus.GlobalId, { id: oskar.GlobalId, share: 1 / 4 }],
  }
);

About

a feature-complete typescript client for the Splid API.

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published