Skip to content

Methods and Events

This section of SDK covers the topic related to apps communication.

Calling Methods

To call the Telegram Mini Apps methods, developer should use postEvent function:

typescript
import { postEvent } from '@tma.js/sdk';

postEvent('web_app_setup_back_button', { is_visible: true });
import { postEvent } from '@tma.js/sdk';

postEvent('web_app_setup_back_button', { is_visible: true });

This function automatically finds the correct way to send this event based on the current environment features. For greater accuracy, it determines the current Telegram application type and selects the appropriate flow.

request

request function should be used in case, it is required to call some Telegram Mini Apps method and receive specified event. For example, developer would like to call web_app_request_viewport method and catch viewport_changed event, to receive actual viewport data.

typescript
import { request } from '@tma.js/sdk';

const viewport = await request(
  'web_app_request_viewport',
  'viewport_changed'
);

console.log(viewport);
// Output:
// {
//   is_state_stable: true,
//   is_expanded: false,
//   height: 320
// };
import { request } from '@tma.js/sdk';

const viewport = await request(
  'web_app_request_viewport',
  'viewport_changed'
);

console.log(viewport);
// Output:
// {
//   is_state_stable: true,
//   is_expanded: false,
//   height: 320
// };

In case, Telegram Mini Apps method accepts parameters, you should pass them in the params property of the third argument:

typescript
import { request } from '@tma.js/sdk';

const buttonId = await request('web_app_open_popup', 'popup_closed', {
  params: {
    title: 'Caution',
    message: 'Should we delete you account?',
    buttons: [
      { id: 'yes', type: 'ok' },
      { id: 'no', type: 'cancel' },
    ],
  },
});
import { request } from '@tma.js/sdk';

const buttonId = await request('web_app_open_popup', 'popup_closed', {
  params: {
    title: 'Caution',
    message: 'Should we delete you account?',
    buttons: [
      { id: 'yes', type: 'ok' },
      { id: 'no', type: 'cancel' },
    ],
  },
});

Alternatively, you could pass the object describing all options:

typescript
import { request } from '@tma.js/sdk';

const buttonId = await request({
  method: 'web_app_open_popup',
  event: 'popup_closed',
  params: {
    title: 'Caution',
    message: 'Should we delete you account?',
    buttons: [
      { id: 'yes', type: 'ok' },
      { id: 'no', type: 'cancel' },
    ],
  },
});
import { request } from '@tma.js/sdk';

const buttonId = await request({
  method: 'web_app_open_popup',
  event: 'popup_closed',
  params: {
    title: 'Caution',
    message: 'Should we delete you account?',
    buttons: [
      { id: 'yes', type: 'ok' },
      { id: 'no', type: 'cancel' },
    ],
  },
});

Moreover, you can track several events in the same time:

typescript
import { request } from '@tma.js/sdk';

const result = await request({
  method: 'web_app_open_scan_qr_popup',
  event: ['qr_text_received', 'scan_qr_popup_closed'],
});

// result will either be qr_text_received or 
// scan_qr_popup_closed events payload.
import { request } from '@tma.js/sdk';

const result = await request({
  method: 'web_app_open_scan_qr_popup',
  event: ['qr_text_received', 'scan_qr_popup_closed'],
});

// result will either be qr_text_received or 
// scan_qr_popup_closed events payload.

This function allows passing additional options, such as postEvent, timeout and capture.

postEvent

We use postEvent option to override the method, which is used to call the Telegram Mini Apps method.

typescript
import { request, createPostEvent } from '@tma.js/sdk';

request('web_app_request_viewport', 'viewport_changed', {
  postEvent: createPostEvent('6.5'),
});
import { request, createPostEvent } from '@tma.js/sdk';

request('web_app_request_viewport', 'viewport_changed', {
  postEvent: createPostEvent('6.5'),
});

timeout

timeout option is responsible for assigning the request timeout. In case, timeout was reached, an error will be thrown.

typescript
import { request, isTimeoutError } from '@tma.js/sdk';

try {
  await request(
    'web_app_invoke_custom_method',
    'custom_method_invoked',
    {
      timeout: 5000,
      params: {
        req_id: '1',
        method: 'deleteStorageValues',
        params: { keys: ['a'] },
      },
    },
  );
} catch (e) {
  if (isTimeoutError(e)) {
    console.error('Timeout error:', e);
    return;
  }
  console.error('Some different error', e);
}
import { request, isTimeoutError } from '@tma.js/sdk';

try {
  await request(
    'web_app_invoke_custom_method',
    'custom_method_invoked',
    {
      timeout: 5000,
      params: {
        req_id: '1',
        method: 'deleteStorageValues',
        params: { keys: ['a'] },
      },
    },
  );
} catch (e) {
  if (isTimeoutError(e)) {
    console.error('Timeout error:', e);
    return;
  }
  console.error('Some different error', e);
}

capture

capture property is a function, that allows developer to determine if occurred Telegram Mini Apps event should be captured and returned from the request function:

typescript
const slug = 'jjKSJnm1k23lodd';

request('web_app_open_invoice', 'invoice_closed', {
  params: { slug },
  capture(data) {
    return slug === data.slug;
  },
});
const slug = 'jjKSJnm1k23lodd';

request('web_app_open_invoice', 'invoice_closed', {
  params: { slug },
  capture(data) {
    return slug === data.slug;
  },
});

By default, the request function captures the first event with required name. In this case, request function will capture the event only in case, it has the expected slug.

Invoking Custom Methods

Custom methods are methods, which could be used by Telegram Mini Apps web_app_invoke_custom_method method. invokeCustomMethod function simplifies usage of such methods and reuses the request function.

Here is the code example without using this function:

typescript
import { request } from '@tma.js/sdk';

const reqId = 'ABC';

request('web_app_invoke_custom_method', 'custom_method_invoked', {
  params: {
    req_id: reqId,
    method: 'deleteStorageValues',
    params: { keys: ['a'] },
  },
  capture(data) {
    return data.req_id === reqId;
  }
});
import { request } from '@tma.js/sdk';

const reqId = 'ABC';

request('web_app_invoke_custom_method', 'custom_method_invoked', {
  params: {
    req_id: reqId,
    method: 'deleteStorageValues',
    params: { keys: ['a'] },
  },
  capture(data) {
    return data.req_id === reqId;
  }
});

And that is how we could rewrite it using the invokeCustomMethod function:

typescript
import { invokeCustomMethod } from '@tma.js/sdk';

invokeCustomMethod('deleteStorageValues', { keys: ['a'] }, 'ABC');
import { invokeCustomMethod } from '@tma.js/sdk';

invokeCustomMethod('deleteStorageValues', { keys: ['a'] }, 'ABC');

In contrary to the request function, the invokeCustomMethod function parses the result and checks if contains the error property. In case it does, the function will throw the according error. Otherwise, the result property will be returned.

Listening to Events

on and off

To start working with events, developer could use on and off functions. Here is how basic on function usage looks like:

typescript
import { on } from '@tma.js/sdk';

// Start listening to "viewport_changed" event. Returned value
// is a function, which removes this event listener.
const removeListener = on('viewport_changed', payload => {
  console.log('Viewport changed:', payload);
});

// Remove this event listener.
removeListener();
import { on } from '@tma.js/sdk';

// Start listening to "viewport_changed" event. Returned value
// is a function, which removes this event listener.
const removeListener = on('viewport_changed', payload => {
  console.log('Viewport changed:', payload);
});

// Remove this event listener.
removeListener();

To stop listening to events, developer could alternatively use off function:

typescript
import { on, off, type MiniAppsEventListener } from '@tma.js/sdk';

const listener: MiniAppsEventListener<'viewport_changed'> = payload => {
  console.log('Viewport changed:', payload);
};

// Start listening to event.
on('viewport_changed', listener);

// Remove event listener.
off('viewport_changed', listener);
import { on, off, type MiniAppsEventListener } from '@tma.js/sdk';

const listener: MiniAppsEventListener<'viewport_changed'> = payload => {
  console.log('Viewport changed:', payload);
};

// Start listening to event.
on('viewport_changed', listener);

// Remove event listener.
off('viewport_changed', listener);

subscribe and unsubscribe

To listen to all events sent from the native Telegram application, developer should utilize such functions as subscribe and unsubscribe:

typescript
import {
  subscribe,
  unsubscribe,
  type MiniAppsGlobalEventListener,
} from '@tma.js/sdk';

const listener: MiniAppsGlobalEventListener = (event, data) => {
  console.log('Received event', event, 'with data', data);
};

// Listen to all events.
subscribe(listener);

// Remove this listener.
unsubscribe(listener);
import {
  subscribe,
  unsubscribe,
  type MiniAppsGlobalEventListener,
} from '@tma.js/sdk';

const listener: MiniAppsGlobalEventListener = (event, data) => {
  console.log('Received event', event, 'with data', data);
};

// Listen to all events.
subscribe(listener);

// Remove this listener.
unsubscribe(listener);

Checking Method Support

postEvent function itself is not checking if specified method supported by current native Telegram application. To do this, developer could use supports function which accepts Telegram Mini Apps method name and current platform version:

typescript
import { supports } from '@tma.js/sdk';

supports('web_app_trigger_haptic_feedback', '6.0'); // false
supports('web_app_trigger_haptic_feedback', '6.1'); // true
import { supports } from '@tma.js/sdk';

supports('web_app_trigger_haptic_feedback', '6.0'); // false
supports('web_app_trigger_haptic_feedback', '6.1'); // true

The supports function also allows checking if specified parameter in method parameters is supported:

typescript
import { supports } from '@tma.js/sdk';

supports('web_app_open_link', 'try_instant_view', '6.0'); // false
supports('web_app_open_link', 'try_instant_view', '6.7'); // true
import { supports } from '@tma.js/sdk';

supports('web_app_open_link', 'try_instant_view', '6.0'); // false
supports('web_app_open_link', 'try_instant_view', '6.7'); // true

TIP

It is recommended to use this function before calling Telegram Mini Apps methods to prevent applications from stalling and other unexpected behavior.

Creating safer postEvent

This package includes a function named createPostEvent that takes the current Telegram Mini Apps version as input. It returns the postEvent function, which internally checks if the specified method and parameters are supported. If they are not, the function will throw an error.

typescript
import { createPostEvent } from '@tma.js/sdk';

const postEvent = createPostEvent('6.5');

// Will work fine.
postEvent('web_app_read_text_from_clipboard');

// Will throw an error.
postEvent('web_app_request_phone');
import { createPostEvent } from '@tma.js/sdk';

const postEvent = createPostEvent('6.5');

// Will work fine.
postEvent('web_app_read_text_from_clipboard');

// Will throw an error.
postEvent('web_app_request_phone');

It is highly recommended to use this postEvent generator to ensure that method calls work as expected.

Debugging

Package supports enabling the debug mode, which leads to logging messages related to events handling. To change debug mode, use setDebug function:

typescript
import { setDebug } from '@tma.js/sdk';

setDebug(true);
import { setDebug } from '@tma.js/sdk';

setDebug(true);

Target Origin

If the package is being used in a browser environment (iframe), packages employs the function window.parent.postMessage. This function requires specifying the target origin to ensure events are only sent to trusted parent iframes. By default, the package utilizes https://web.telegram.org as the origin. To enable event transmission to other origins, developer should utilize the setTargetOrigin function:

typescript
import { setTargetOrigin } from '@tma.js/sdk';

setTargetOrigin('https://myendpoint.org');
import { setTargetOrigin } from '@tma.js/sdk';

setTargetOrigin('https://myendpoint.org');

WARNING

It is strongly recommended not to override this value as long as it could lead to security issues. Specify this value only for test purposes.

Released under the MIT License.