UNPKG

react-url-query

Version:

A library for managing state through query parameters in the URL in React. Works well with or without Redux and React Router.

420 lines (367 loc) 10.3 kB
/** * Functions for encoding and decoding values as strings. */ import config from './urlQueryConfig'; /** * Encodes a date as a string in YYYY-MM-DD format. * * @param {Date} date * @return {String} the encoded date */ export function encodeDate(date) { if (date == null) { return date; } const year = date.getFullYear(); const month = date.getMonth() + 1; const day = date.getDate(); return `${year}-${month < 10 ? `0${month}` : month}-${ day < 10 ? `0${day}` : day }`; } /** * Converts a date in the format 'YYYY-mm-dd...' into a proper date, because * new Date() does not do that correctly. The date can be as complete or incomplete * as necessary (aka, '2015', '2015-10', '2015-10-01'). * It will not work for dates that have times included in them. * * @param {String} dateString String date form like '2015-10-01' * @return {Date} parsed date */ export function decodeDate(dateString) { if (dateString == null || !dateString.length) { return undefined; } const parts = dateString.split('-'); // may only be a year so won't even have a month if (parts[1] != null) { parts[1] -= 1; // Note: months are 0-based } else { // just a year, set the month and day to the first parts[1] = 0; parts[2] = 1; } const decoded = new Date(...parts); if (isNaN(decoded.getTime())) { return undefined; } return decoded; } /** * Encodes a boolean as a string. true -> "1", false -> "0". * * @param {Boolean} bool * @return {String} the encoded boolean */ export function encodeBoolean(bool) { if (bool === undefined) { return undefined; } return bool ? '1' : '0'; } /** * Decodes a boolean from a string. "1" -> true, "0" -> false. * Everything else maps to undefined. * * @param {String} boolStr the encoded boolean string * @return {Boolean} the boolean value */ export function decodeBoolean(boolStr) { if (boolStr === '1') { return true; } else if (boolStr === '0') { return false; } return undefined; } /** * Encodes a number as a string. * * @param {Number} num * @return {String} the encoded number */ export function encodeNumber(num) { if (num == null) { return undefined; } return String(num); } /** * Decodes a number from a string via parseFloat. If the number is invalid, * it returns undefined. * * @param {String} numStr the encoded number string * @return {Number} the number value */ export function decodeNumber(numStr) { if (numStr == null) { return undefined; } const result = parseFloat(numStr); if (isNaN(result)) { return undefined; } return result; } /** * Encodes a string while safely handling null and undefined values. * * @param {String} string * @return {String} the encoded string */ export function encodeString(str) { if (str == null) { return undefined; } return String(str); } /** * Decodes a string while safely handling null and undefined values. * * @param {String} str the encoded string * @return {String} the string value */ export function decodeString(str) { if (str == null) { return undefined; } return String(str); } /** * Encodes anything as a JSON string. * * @param {Any} any The thing to be encoded * @return {String} The JSON string representation of any */ export function encodeJson(any) { if (any == null) { return undefined; } return JSON.stringify(any); } /** * Decodes a JSON string into javascript * * @param {String} jsonStr The JSON string representation * @return {Any} The javascript representation */ export function decodeJson(jsonStr) { if (!jsonStr) { return undefined; } let result; try { result = JSON.parse(jsonStr); } catch (e) { /* ignore errors, returning undefined */ } return result; } /** * Encodes an array as a JSON string. * * @param {Array} array The array to be encoded * @return {String} The JSON string representation of array */ export function encodeArray(array, entrySeparator = config.entrySeparator) { if (!array) { return undefined; } return array.join(entrySeparator); } /** * Decodes a JSON string into javascript array * * @param {String} jsonStr The JSON string representation * @return {Array} The javascript representation */ export function decodeArray(arrayStr, entrySeparator = config.entrySeparator) { if (!arrayStr) { return undefined; } return arrayStr .split(entrySeparator) .map(item => (item === '' ? undefined : item)); } /** * Encodes a numeric array as a JSON string. (alias of encodeArray) * * @param {Array} array The array to be encoded * @return {String} The JSON string representation of array */ export const encodeNumericArray = encodeArray; /** * Decodes a JSON string into javascript array where all entries are numbers * * @param {String} jsonStr The JSON string representation * @return {Array} The javascript representation */ export function decodeNumericArray( arrayStr, entrySeparator = config.entrySeparator ) { const decoded = decodeArray(arrayStr, entrySeparator); if (!decoded) { return undefined; } return decoded.map(d => (d == null ? d : +d)); } /** * Encode simple objects as readable strings. Currently works only for simple, * flat objects where values are numbers, booleans or strings. * * For example { foo: bar, boo: baz } -> "foo-bar_boo-baz" * * @param {Object} object The object to encode * @param {String} keyValSeparator="-" The separator between keys and values * @param {String} entrySeparator="_" The separator between entries * @return {String} The encoded object */ export function encodeObject( obj, keyValSeparator = config.keyValSeparator, entrySeparator = config.entrySeparator ) { if (!obj || !Object.keys(obj).length) { return undefined; } return Object.keys(obj) .map(key => `${key}${keyValSeparator}${obj[key]}`) .join(entrySeparator); } /** * Decodes a simple object to javascript. Currently works only for simple, * flat objects where values are numbers, booleans or strings. * * For example "foo-bar_boo-baz" -> { foo: bar, boo: baz } * * @param {String} objStr The object string to decode * @param {String} keyValSeparator="-" The separator between keys and values * @param {String} entrySeparator="_" The separator between entries * @return {Object} The javascript object */ export function decodeObject( objStr, keyValSeparator = config.keyValSeparator, entrySeparator = config.entrySeparator ) { if (!objStr || !objStr.length) { return undefined; } const obj = {}; objStr.split(entrySeparator).forEach(entryStr => { const [key, value] = entryStr.split(keyValSeparator); obj[key] = value; }); return obj; } /** * Encode simple objects as readable strings. Alias of encodeObject. * * For example { foo: 123, boo: 521 } -> "foo-123_boo-521" * * @param {Object} object The object to encode * @param {String} keyValSeparator="-" The separator between keys and values * @param {String} entrySeparator="_" The separator between entries * @return {String} The encoded object */ export const encodeNumericObject = encodeObject; /** * Decodes a simple object to javascript where all values are numbers. * Currently works only for simple, flat objects. * * For example "foo-123_boo-521" -> { foo: 123, boo: 521 } * * @param {String} objStr The object string to decode * @param {String} keyValSeparator="-" The separator between keys and values * @param {String} entrySeparator="_" The separator between entries * @return {Object} The javascript object */ export function decodeNumericObject( objStr, keyValSeparator = config.keyValSeparator, entrySeparator = config.entrySeparator ) { const decoded = decodeObject(objStr, keyValSeparator, entrySeparator); if (!decoded) { return undefined; } // convert to numbers Object.keys(decoded).forEach(key => { decoded[key] = decoded[key] == null ? decoded[key] : +decoded[key]; }); return decoded; } /** * Collection of Decoders by type */ export const Decoders = { number: decodeNumber, string: decodeString, object: decodeObject, array: decodeArray, json: decodeJson, date: decodeDate, boolean: decodeBoolean, numericObject: decodeNumericObject, numericArray: decodeNumericArray, }; /** * Generic decode function that takes a type as an argument. * * @param {String|Function} type If a function, it is used to decode, otherwise the string is * the key for the decoder in the Decoders collection. * @param {String} encodedValue the value to decode * @param {Any} defaultValue The default value to use if encodedValue is undefined. * @return {Any} The decoded value */ export function decode(type, encodedValue, defaultValue) { let decodedValue; if (typeof type === 'function') { decodedValue = type(encodedValue, defaultValue); } else if (typeof type === 'object' && type.decode) { decodedValue = type.decode(encodedValue, defaultValue); } else if (encodedValue === undefined) { decodedValue = defaultValue; } else if (Decoders[type]) { decodedValue = Decoders[type](encodedValue); } else { decodedValue = encodedValue; } return decodedValue; } /** * Collection of Encoders by type */ export const Encoders = { number: encodeNumber, string: encodeString, object: encodeObject, array: encodeArray, json: encodeJson, date: encodeDate, boolean: encodeBoolean, numericObject: encodeNumericObject, numericArray: encodeNumericArray, }; /** * Generic encode function that takes a type as an argument. * * @param {String|Function} type If a function, it is used to encode, otherwise the string is * the key for the encoder in the Encoders collection. * @param {String} decodedValue the value to encode * @return {Any} The encoded value */ export function encode(type, decodedValue) { let encodedValue; if (typeof type === 'function') { encodedValue = type(decodedValue); } else if (typeof type === 'object' && type.encode) { encodedValue = type.encode(decodedValue); } else if (Encoders[type]) { encodedValue = Encoders[type](decodedValue); } else { encodedValue = decodedValue; } return encodedValue; }