
Sagar Deep Kushwaha
Mastering JavaScript Promise Polyfill: A Complete Guide with Examples

Chapters
- Understanding Promise Polyfills
- The Importance of Promise Polyfills
- Basic functionality of Promise API
- Implementing a Promise Polyfill
- Example promise polyfill
- Conclusion
Understanding Promise Polyfills
A Promise polyfill is a JavaScript code snippet designed to fill the gap left by browsers that lack native support for Promises. These polyfills replicate the Promise API, enabling developers to leverage Promises in environments where they are not natively available.
The Importance of Promise Polyfills
In the ever-evolving web ecosystem, ensuring cross-browser compatibility is paramount. By employing Promise polyfills, developers can write code using modern asynchronous patterns without worrying about compatibility issues in older browsers. This approach leads to cleaner, more maintainable codebases and enhances the user experience across diverse browsing environments.
Basic functionality of Promise API
The Promise API in JavaScript provides a set of methods and functionality for handling asynchronous operations. Here are the key functionalities supported by the Promise API:
- Creating Promises object
- Chaining Promises
- Error handling (Reject function support)
- On fulfill (Resolver function support)
- Helper functions, eg. Promise.all, Promise.any, Promise.race ....
Traditional way of creating and using promise
const Executor = ((resolve, reject) => { //async task setTimeout(() => { resolve(2); },1000); //if something went wrong => call reject }) const myPromise = new Promise(Executor); myPromise.then(ans => {console.log(ans)}).catch(e =>handleError(e));
Promise states
- Promise Fulfilled : meaning that the operation was completed successfully.
- Promise Rejected : meaning that the operation failed.
- Promise Pending : initial state, neither fulfilled nor rejected.
- Promise Settled : Either fulfilled or rejected
Implementing a Promise Polyfill
Let's explore a simplified example of a Promise polyfill:
In this example, we create a basic implementation of a Promise polyfill that supports the then and catch methods.
1. Creating the Promise Constructor
The Promise constructor is the foundation of the polyfill. It accepts an executor function, which in turn receives resolve and reject functions as arguments. Inside this executor function, the asynchronous operation is performed, and based on its outcome, either resolve or reject is called.
if (!window.Promise) { //checks if Promise object is available or not in global object window.Promise = function (executor) { // Polyfill implementation }; }
2. Handling Promise State and Values
The polyfill needs to maintain the state of the Promise (pending, fulfilled, or rejected) and store the value or reason when the Promise settles.
window.Promise = function (executor) { var ref= this; ref.state = 'pending'; ref.value = undefined; ref.callbacks = []; // executor function and state transitions };
3. Implementing Resolve and Reject
Inside the executor function, resolve and reject are defined. These functions transition the Promise's state to either fulfilled or rejected and store the corresponding value or reason. Additionally, they execute any registered callback functions.
window.Promise = function (executor) { // Other code... /* Resolve function is called by executor when the asynchronous task is finished. like in the uses example : we are calling the resolve after waiting for 1000 ms. */ function resolve(value) { // if (ref.state !== 'pending') return; ref.state = 'fulfilled'; ref.value = value; ref.callbacks.forEach(function (callback) { return callback.onFulfilled(value); }); } /* Resolve function is called by executor when the asynchronous task has thrown error. */ function reject(reason) { if (ref.state !== 'pending') return; ref.state = 'rejected'; ref.value = reason; ref.callbacks.forEach(function (callback) { return callback.onRejected(reason); }); } //Whenever we define a Promise object the executor function is called immediately executor(resolve, reject); };
4. Chaining Promises with then()
The then method is crucial for chaining Promises. It registers callback functions to be executed when the Promise is fulfilled or rejected. This involves handling different scenarios based on the Promise's state at the time thenis called.
window.Promise.prototype.then = function (onFulfilled, onRejected) { var ref= this; return new Promise(function (resolve, reject) { // Handling different states and executing callbacks }); };
5. Handling Errors with catch()
The catch method is used to handle errors that occur within the Promise chain. It acts as a shorthand for calling then(undefined, onRejected). This ensures consistent error handling throughout the Promise chain.
window.Promise.prototype.catch = function (onRejected) { return this.then(null, onRejected); };
Example promise polyfill
if (!window.Promise) { window.Promise = function (executor) { var ref= this; ref.state = 'pending'; ref.value = undefined; ref.callbacks = []; function resolve(value) { if (ref.state !== 'pending') return; ref.state = 'fulfilled'; ref.value = value; ref.callbacks.forEach(function (callback) { return callback.onFulfilled(value); }); } function reject(reason) { if (ref.state !== 'pending') return; ref.state = 'rejected'; ref.value = reason; ref.callbacks.forEach(function (callback) { return callback.onRejected(reason); }); } executor(resolve, reject); }; window.Promise.prototype.then = function (onFulfilled, onRejected) { var ref= this; return new Promise(function (resolve, reject) { function handle(callback) { try { var result = callback(ref.value); if (result instanceof Promise) { result.then(resolve, reject); } else { resolve(result); } } catch (error) { reject(error); } } if (ref.state === 'fulfilled') { handle(onFulfilled); } else if (ref.state === 'rejected') { handle(onRejected); } else { ref.callbacks.push({ onFulfilled: function onFulfilled(value) { return handle(onFulfilled); }, onRejected: function onRejected(reason) { return handle(onRejected); }, }); } }); }; window.Promise.prototype.catch = function (onRejected) { return this.then(null, onRejected); }; }```
Conclusion
Promise polyfills play a crucial role in ensuring the compatibility of modern JavaScript code across different browsers. By understanding how Promise polyfills work and implementing them effectively, developers can maintain consistency and reliability in their applications, regardless of the browser environment. With this knowledge, you can confidently leverage Promises in your projects, knowing that they will function reliably across all platforms.
More articles
Dive into the world of TypeScript, the superset of JavaScript that brings static typing and advanced features to web development.
Are you gearing up for a JavaScript interview and feeling the nerves creeping in? Don't worry, you're not alone. Interviews can be intimidating, but with the right preparation, you can ace them with confidence. JavaScript interviews. Let's delve into the top 6 must know coding questions you must revise before stepping into that interview room.
Chapters
- Understanding Promise Polyfills
- The Importance of Promise Polyfills
- Basic functionality of Promise API
- Implementing a Promise Polyfill
- Example promise polyfill
- Conclusion
Top articles
Demystifying TypeScript: Exploring Its Core Concepts and Features
Dive into the world of TypeScript, the superset of JavaScript that brings static typing and advanced features to web development.
6 Must Revise Javascript coding interview questions before any interview
Are you gearing up for a JavaScript interview and feeling the nerves creeping in? Don't worry, you're not alone. Interviews can be intimidating, but with the right preparation, you can ace them with confidence. JavaScript interviews. Let's delve into the top 6 must know coding questions you must revise before stepping into that interview room.