@kwiz/common 1.0.78 → 1.0.80
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.github/workflows/npm-publish.yml +24 -0
- package/.madgerc +2 -2
- package/LICENSE +21 -21
- package/fix-folder-imports.js +26 -26
- package/lib/cjs/helpers/sharepoint.js +5 -1
- package/lib/cjs/helpers/sharepoint.js.map +1 -1
- package/lib/cjs/helpers/typecheckers.js +5 -1
- package/lib/cjs/helpers/typecheckers.js.map +1 -1
- package/lib/cjs/types/libs/msal.types.js +26 -26
- package/lib/cjs/utils/sharepoint.rest/list.js +1 -1
- package/lib/cjs/utils/sharepoint.rest/list.js.map +1 -1
- package/lib/cjs/utils/sharepoint.rest/user.js +11 -11
- package/lib/esm/helpers/sharepoint.js +3 -0
- package/lib/esm/helpers/sharepoint.js.map +1 -1
- package/lib/esm/helpers/typecheckers.js +3 -0
- package/lib/esm/helpers/typecheckers.js.map +1 -1
- package/lib/esm/types/libs/msal.types.js +26 -26
- package/lib/esm/utils/sharepoint.rest/list.js +2 -2
- package/lib/esm/utils/sharepoint.rest/list.js.map +1 -1
- package/lib/esm/utils/sharepoint.rest/user.js +11 -11
- package/lib/types/helpers/sharepoint.d.ts +1 -0
- package/lib/types/helpers/typecheckers.d.ts +1 -0
- package/package.json +77 -77
- package/readme.md +17 -17
- package/src/_dependencies.ts +12 -12
- package/src/config.ts +17 -17
- package/src/helpers/Guid.ts +181 -181
- package/src/helpers/base64.ts +173 -173
- package/src/helpers/browser.test.js +13 -13
- package/src/helpers/browser.ts +1348 -1348
- package/src/helpers/browserinfo.ts +292 -292
- package/src/helpers/collections.base.test.js +25 -25
- package/src/helpers/collections.base.ts +437 -437
- package/src/helpers/collections.ts +107 -107
- package/src/helpers/color.ts +54 -54
- package/src/helpers/cookies.ts +59 -59
- package/src/helpers/date.test.js +119 -119
- package/src/helpers/date.ts +188 -188
- package/src/helpers/debug.ts +186 -186
- package/src/helpers/emails.ts +6 -6
- package/src/helpers/eval.ts +5 -5
- package/src/helpers/file.test.js +50 -50
- package/src/helpers/file.ts +58 -58
- package/src/helpers/flatted.ts +149 -149
- package/src/helpers/functions.ts +16 -16
- package/src/helpers/graph/calendar.types.ts +10 -10
- package/src/helpers/http.ts +69 -69
- package/src/helpers/images.ts +22 -22
- package/src/helpers/json.ts +38 -38
- package/src/helpers/md5.ts +189 -189
- package/src/helpers/objects.test.js +33 -33
- package/src/helpers/objects.ts +270 -270
- package/src/helpers/promises.test.js +37 -37
- package/src/helpers/promises.ts +165 -165
- package/src/helpers/random.ts +27 -27
- package/src/helpers/scheduler/scheduler.test.js +103 -103
- package/src/helpers/scheduler/scheduler.ts +131 -131
- package/src/helpers/sharepoint.ts +776 -772
- package/src/helpers/strings.test.js +101 -101
- package/src/helpers/strings.ts +317 -317
- package/src/helpers/typecheckers.test.js +34 -34
- package/src/helpers/typecheckers.ts +266 -262
- package/src/helpers/url.test.js +43 -43
- package/src/helpers/url.ts +207 -207
- package/src/helpers/urlhelper.ts +111 -111
- package/src/index.ts +6 -6
- package/src/types/auth.ts +54 -54
- package/src/types/common.types.ts +15 -15
- package/src/types/flatted.types.ts +59 -59
- package/src/types/globals.types.ts +6 -6
- package/src/types/graph/calendar.types.ts +80 -80
- package/src/types/knownscript.types.ts +18 -18
- package/src/types/libs/datajs.types.ts +28 -28
- package/src/types/libs/ics.types.ts +30 -30
- package/src/types/libs/msal.types.ts +49 -49
- package/src/types/locales.ts +124 -124
- package/src/types/localstoragecache.types.ts +8 -8
- package/src/types/location.types.ts +27 -27
- package/src/types/moment.ts +11 -11
- package/src/types/regex.types.ts +16 -16
- package/src/types/rest.types.ts +95 -95
- package/src/types/sharepoint.types.ts +1465 -1465
- package/src/types/sharepoint.utils.types.ts +287 -287
- package/src/utils/auth/common.ts +74 -74
- package/src/utils/auth/discovery.test.js +12 -12
- package/src/utils/auth/discovery.ts +132 -132
- package/src/utils/base64.ts +27 -27
- package/src/utils/consolelogger.ts +320 -320
- package/src/utils/date.ts +35 -35
- package/src/utils/emails.ts +24 -24
- package/src/utils/knownscript.ts +286 -286
- package/src/utils/localstoragecache.ts +441 -441
- package/src/utils/rest.ts +501 -501
- package/src/utils/script.ts +170 -170
- package/src/utils/sharepoint.rest/common.ts +154 -154
- package/src/utils/sharepoint.rest/date.ts +62 -62
- package/src/utils/sharepoint.rest/file.folder.ts +598 -598
- package/src/utils/sharepoint.rest/item.ts +547 -547
- package/src/utils/sharepoint.rest/list.ts +1388 -1388
- package/src/utils/sharepoint.rest/listutils/GetListItemsByCaml.ts +774 -774
- package/src/utils/sharepoint.rest/listutils/GetListItemsById.ts +275 -275
- package/src/utils/sharepoint.rest/listutils/common.ts +206 -206
- package/src/utils/sharepoint.rest/location.ts +141 -141
- package/src/utils/sharepoint.rest/navigation-links.ts +86 -86
- package/src/utils/sharepoint.rest/user-search.ts +252 -252
- package/src/utils/sharepoint.rest/user.ts +491 -491
- package/src/utils/sharepoint.rest/web.ts +1384 -1384
- package/src/utils/sod.ts +194 -194
- package/lib/cjs/helpers/_dependencies.js +0 -21
- package/lib/cjs/helpers/_dependencies.js.map +0 -1
- package/lib/cjs/utils/_dependencies.js +0 -24
- package/lib/cjs/utils/_dependencies.js.map +0 -1
- package/lib/esm/helpers/_dependencies.js +0 -3
- package/lib/esm/helpers/_dependencies.js.map +0 -1
- package/lib/esm/utils/_dependencies.js +0 -4
- package/lib/esm/utils/_dependencies.js.map +0 -1
- package/lib/types/helpers/_dependencies.d.ts +0 -2
- package/lib/types/utils/_dependencies.d.ts +0 -3
package/src/helpers/strings.ts
CHANGED
|
@@ -1,318 +1,318 @@
|
|
|
1
|
-
import { IDictionary } from "../types/common.types";
|
|
2
|
-
import { makeUniqueArray } from "./collections.base";
|
|
3
|
-
import { isNullOrEmptyString, isNullOrUndefined, isNumber, isString } from "./typecheckers";
|
|
4
|
-
|
|
5
|
-
export function endsWith(str: string, value: string, ignoreCase?: boolean): boolean {
|
|
6
|
-
let str1 = str;
|
|
7
|
-
let find = value;
|
|
8
|
-
if (ignoreCase) {
|
|
9
|
-
str1 = str1.toLowerCase();
|
|
10
|
-
find = find.toLowerCase();
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
return str1.substr(str1.length - find.length) === find;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
export function startsWith(str: string, value: string, ignoreCase?: boolean): boolean {
|
|
17
|
-
let str1 = str;
|
|
18
|
-
let find = value;
|
|
19
|
-
if (ignoreCase) {
|
|
20
|
-
str1 = str1.toLowerCase();
|
|
21
|
-
find = find.toLowerCase();
|
|
22
|
-
}
|
|
23
|
-
return str1.substr(0, find.length) === find;
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
/** remove space at start or end */
|
|
27
|
-
export function trim(str: string): string {
|
|
28
|
-
return str.replace(/^\s+|\s+$/g, '');
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
export function trimEnd(str: string): string {
|
|
32
|
-
return str.replace(/\s+$/, "");
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
export function trimStart(str: string): string {
|
|
36
|
-
return str.replace(/^\s+/, "");
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
export function splice(str: string, start: number, delCount: number, insert: string) {
|
|
40
|
-
return str.slice(0, start) + insert + str.slice(start + Math.abs(delCount));
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
export function padRight(value: string | number, length: number, fillString: string = "0") {
|
|
44
|
-
if (isNumber(value))
|
|
45
|
-
value = value.toString(10);
|
|
46
|
-
let pad = isNullOrEmptyString(fillString) ? "0" : fillString[0];
|
|
47
|
-
return value + Array(length - value.length + 1).join(pad);
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
export function padLeft(value: string | number, length: number, fillString: string = "0") {
|
|
51
|
-
if (isNumber(value))
|
|
52
|
-
value = value.toString(10);
|
|
53
|
-
let pad = isNullOrEmptyString(fillString) ? "0" : fillString[0];
|
|
54
|
-
return Array(length - String(value).length + 1).join(pad) + value;
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
/** returns array of [token] found inside the string
|
|
58
|
-
* supports token formats [Author??Created by {0}::Unknown author] will return "Author" as the token
|
|
59
|
-
* */
|
|
60
|
-
export function GetTokens(StringFormat: string): string[] {
|
|
61
|
-
let tokensResult: string[] = [];
|
|
62
|
-
if (isNullOrEmptyString(StringFormat)) return tokensResult;
|
|
63
|
-
|
|
64
|
-
let tokens = StringFormat.match(/\[[^\]]*\]/g);
|
|
65
|
-
if (tokens && tokens.length > 0) {
|
|
66
|
-
tokens.forEach(token => {
|
|
67
|
-
let key = token.slice(1, token.length - 1);
|
|
68
|
-
key = GetTokenInfo(key).tokenName;
|
|
69
|
-
if (tokensResult.indexOf(key) < 0)
|
|
70
|
-
tokensResult.push(key);
|
|
71
|
-
});
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
return tokensResult;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
/** replaces a string with [token] and [otherToken] with their matched provided values
|
|
78
|
-
* supports token formats [Author??Created by {0}::Unknown author]
|
|
79
|
-
*/
|
|
80
|
-
export function ReplaceTokens(StringFormat: string, TokenValues: IDictionary<string>, options?: {
|
|
81
|
-
/**set to true if you want to keep "[token]" in the string when a token value wasn't provided */
|
|
82
|
-
keepMissingTokens?: boolean;
|
|
83
|
-
}): string {
|
|
84
|
-
|
|
85
|
-
let skipMissingTokens = options && options.keepMissingTokens;
|
|
86
|
-
|
|
87
|
-
if (isNullOrUndefined(StringFormat)) return null;
|
|
88
|
-
if (StringFormat !== '') {
|
|
89
|
-
let tokens = StringFormat.match(/\[[^\]]*\]/g);
|
|
90
|
-
if (tokens && tokens.length > 0) {
|
|
91
|
-
if (isNullOrUndefined(TokenValues)) TokenValues = {};
|
|
92
|
-
|
|
93
|
-
tokens.forEach(token => {
|
|
94
|
-
let key = token.slice(1, token.length - 1);
|
|
95
|
-
let tokenInfo = GetTokenInfo(key);
|
|
96
|
-
let value = TokenValues[tokenInfo.tokenName];
|
|
97
|
-
let skip = false;
|
|
98
|
-
if (isNullOrUndefined(value)) {
|
|
99
|
-
value = "";
|
|
100
|
-
skip = skipMissingTokens;//if true we won't replace this one
|
|
101
|
-
}
|
|
102
|
-
if (!skip || tokenInfo.hasFormat)
|
|
103
|
-
StringFormat = StringFormat.replace(token, tokenInfo.getValue(value));
|
|
104
|
-
});
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
return StringFormat;
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
/** calls replace tokens on every dictionary value, or set null keys to "". This function ignores null/empty dictionaries */
|
|
111
|
-
export function ReplaceTokensInDictionary(Dictionary: IDictionary<string>, TokenValues: IDictionary<string>) {
|
|
112
|
-
if (!isNullOrUndefined(Dictionary)) {
|
|
113
|
-
Object.keys(Dictionary).forEach(key => {
|
|
114
|
-
Dictionary[key] = isNullOrEmptyString(Dictionary[key]) ? "" : ReplaceTokens(Dictionary[key], TokenValues);
|
|
115
|
-
});
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
/** Normalizes a guid string, lower case and removes {} */
|
|
120
|
-
export function normalizeGuid(text: string, removeDashes?: boolean): string {
|
|
121
|
-
if (isNullOrEmptyString(text) || !isString(text)) {
|
|
122
|
-
return text;
|
|
123
|
-
}
|
|
124
|
-
var guid = text.toLowerCase().trim();
|
|
125
|
-
|
|
126
|
-
if (guid.startsWith("{")) {
|
|
127
|
-
guid = guid.substr(1);
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
if (guid.endsWith("}")) {
|
|
131
|
-
guid = guid.substr(0, guid.length - 1);
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
if (removeDashes) {
|
|
135
|
-
guid = guid.replace(/-/g, '');
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
return guid;
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
export function isEmptyGuid(guid: { toString(): string; }) {
|
|
142
|
-
if (isNullOrEmptyString(guid)) return true;
|
|
143
|
-
else if (Number(normalizeGuid(guid.toString(), true)) === 0) return true;
|
|
144
|
-
return false;
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
export function escapeRegExp(text: string) {
|
|
148
|
-
return text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&');
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
export function isValidDomainLogin(login: string) {
|
|
152
|
-
return /^[A-Za-z0-9\\._-]{7,}$/.test(login);
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
export function stripRichTextWhitespace(value: string) {
|
|
156
|
-
// richText fields in have random markup even when field is empty
|
|
157
|
-
// \u200B zero width space
|
|
158
|
-
// \u200C zero width non-joiner Unicode code point
|
|
159
|
-
// \u200D zero width joiner Unicode code point
|
|
160
|
-
// \uFEFF zero width no-break space Unicode code point
|
|
161
|
-
return isString(value) ? value.replace(/[\u200B-\u200D\uFEFF]/g, "") : value;
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
/** allows min length 1, letters, numbers underscore only */
|
|
165
|
-
export function isValidVarName(text: string) {
|
|
166
|
-
return /^[A-Za-z0-9_]{1,}$/.test(text);
|
|
167
|
-
}
|
|
168
|
-
|
|
169
|
-
/** allows min length 1, letters, numbers underscore and hyphen only */
|
|
170
|
-
export function isValidHeaderName(text: string) {
|
|
171
|
-
return /^[A-Za-z0-9_-]{1,}$/.test(text);
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
/** returns token info with format */
|
|
175
|
-
export function GetTokenInfo(text: string) {
|
|
176
|
-
let split = text.split('??');
|
|
177
|
-
let hasFormat = split.length > 0 && !isNullOrEmptyString(split[1]);
|
|
178
|
-
let formatSplit = hasFormat ? split[1].split('::') : [];
|
|
179
|
-
let valueIfEmpty = formatSplit.length > 1 ? formatSplit[1] : "";
|
|
180
|
-
let formatIfNotEmpty = formatSplit[0];
|
|
181
|
-
let info = {
|
|
182
|
-
tokenName: hasFormat ? split[0] : text,
|
|
183
|
-
hasFormat: hasFormat,
|
|
184
|
-
getValue: (value: string) => {
|
|
185
|
-
if (!hasFormat) return value;
|
|
186
|
-
else {
|
|
187
|
-
if (isNullOrEmptyString(value)) return valueIfEmpty;
|
|
188
|
-
else return formatIfNotEmpty.replace('{0}', value);
|
|
189
|
-
}
|
|
190
|
-
}
|
|
191
|
-
};
|
|
192
|
-
return info;
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
/** return true if both strings are the same, or both are empty/null/undefined */
|
|
196
|
-
export function stringEqualsOrEmpty(str1: string, str2: string, ignoreCase?: boolean) {
|
|
197
|
-
if (isNullOrEmptyString(str1) && isNullOrEmptyString(str2)) return true;
|
|
198
|
-
if (ignoreCase) {
|
|
199
|
-
if (!isNullOrEmptyString(str1)) str1 = str1.toLowerCase();
|
|
200
|
-
if (!isNullOrEmptyString(str2)) str2 = str2.toLowerCase();
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
return str1 === str2;
|
|
204
|
-
}
|
|
205
|
-
|
|
206
|
-
/** return true if str1 contains str2 */
|
|
207
|
-
export function stringContains(str1: string, str2: string, ignoreCase?: boolean) {
|
|
208
|
-
if (isNullOrEmptyString(str1) && isNullOrEmptyString(str2)) return true;
|
|
209
|
-
|
|
210
|
-
if (isNullOrEmptyString(str1))
|
|
211
|
-
str1 = "";
|
|
212
|
-
if (isNullOrEmptyString(str2))
|
|
213
|
-
str2 = "";
|
|
214
|
-
|
|
215
|
-
if (ignoreCase) {
|
|
216
|
-
str1 = str1.toLowerCase();
|
|
217
|
-
str2 = str2.toLowerCase();
|
|
218
|
-
}
|
|
219
|
-
|
|
220
|
-
return str1.indexOf(str2) >= 0;
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
export function cleanupString(str: string, options: {
|
|
224
|
-
replaceNewLines?: string;
|
|
225
|
-
collapseMultipleSpaces?: string;
|
|
226
|
-
collapseMultipleDashes?: string;
|
|
227
|
-
collapseMultipleUnderscore?: string;
|
|
228
|
-
}) {
|
|
229
|
-
if (isString(options.replaceNewLines))
|
|
230
|
-
str = str.replace(/\r/g, '')//no returns
|
|
231
|
-
.replace(/\n/g, options.replaceNewLines);//no line breaks
|
|
232
|
-
if (isString(options.collapseMultipleDashes))
|
|
233
|
-
str = str.replace(/-+/g, options.collapseMultipleSpaces);//no extra spaces
|
|
234
|
-
if (isString(options.collapseMultipleUnderscore))
|
|
235
|
-
str = str.replace(/_+/g, options.collapseMultipleUnderscore);//no extra spaces
|
|
236
|
-
|
|
237
|
-
// do this last, so it will collapse spaces added by previous options
|
|
238
|
-
if (isString(options.collapseMultipleSpaces)) {
|
|
239
|
-
str = str.replace(new RegExp(String.fromCharCode(160), "g"), '')//get rid of non-breaking spaces
|
|
240
|
-
.replace(/ +/g, options.collapseMultipleSpaces);//no extra spaces
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
return str;
|
|
244
|
-
}
|
|
245
|
-
|
|
246
|
-
/** normalizes   to see Issue 752 */
|
|
247
|
-
export function normalizeHtmlSpace(html: string) {
|
|
248
|
-
if (isNullOrEmptyString(html)) return html;
|
|
249
|
-
return html.replace(/ /i, " ");
|
|
250
|
-
}
|
|
251
|
-
|
|
252
|
-
export function replaceAll(str: string, find: string, replace: string, ignoreCase = false) {
|
|
253
|
-
//must call escapeRegExp on find, to make sure it works when there are protected regex characters
|
|
254
|
-
return str.replace(new RegExp(escapeRegExp(find), `g${ignoreCase ? 'i' : ''}`), replace);
|
|
255
|
-
}
|
|
256
|
-
|
|
257
|
-
export function capitalizeFirstLetter(str: string) {
|
|
258
|
-
return isNullOrEmptyString(str)
|
|
259
|
-
? ""
|
|
260
|
-
: `${str.charAt(0).toUpperCase()}${str.substring(1)}`;
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
export function escapeXml(unsafe: string, isAttribute = false) {
|
|
264
|
-
if (isNullOrEmptyString(unsafe)) return "";
|
|
265
|
-
return isAttribute
|
|
266
|
-
? unsafe.replace(/[<>&'"]/g, (c) => {
|
|
267
|
-
switch (c) {
|
|
268
|
-
case '<': return '<';
|
|
269
|
-
case '>': return '>';
|
|
270
|
-
case '&': return '&';
|
|
271
|
-
case '\'': return ''';
|
|
272
|
-
case '"': return '"';
|
|
273
|
-
}
|
|
274
|
-
return c;
|
|
275
|
-
})
|
|
276
|
-
: unsafe.replace(/[<>&]/g, (c) => {
|
|
277
|
-
switch (c) {
|
|
278
|
-
case '<': return '<';
|
|
279
|
-
case '>': return '>';
|
|
280
|
-
case '&': return '&';
|
|
281
|
-
}
|
|
282
|
-
return c;
|
|
283
|
-
});
|
|
284
|
-
}
|
|
285
|
-
|
|
286
|
-
/** uses regex str.match to replace each match by calling the replacer function (imported from CMS) */
|
|
287
|
-
export function replaceRegex(str: string, regex: RegExp, replacer: (match: string) => string | null) {
|
|
288
|
-
let matches = str.match(regex);
|
|
289
|
-
if (!matches || matches.length < 1) return str;
|
|
290
|
-
//replace each found token only once
|
|
291
|
-
let unique = makeUniqueArray(matches);
|
|
292
|
-
//todo: this has a bug where tokens matched contain each other, example, match numbers and prefix with #: 10, 100 will match both and produce #10 ##100
|
|
293
|
-
unique.forEach(m => {
|
|
294
|
-
let replacement = replacer(m);
|
|
295
|
-
if (!isNullOrUndefined(replacement))//ignore nulls
|
|
296
|
-
str = replaceAll(str, m, replacement);
|
|
297
|
-
});
|
|
298
|
-
return str;
|
|
299
|
-
}
|
|
300
|
-
|
|
301
|
-
/** masks a long string, keeping X number for characters at the start/end and replacing the middle with the mask string (default: CC*****CCC) */
|
|
302
|
-
export function maskString(str: string, options?: {
|
|
303
|
-
/** mask string, default ***** */
|
|
304
|
-
mask?: string;
|
|
305
|
-
/** characters to keep at start, default 2 */
|
|
306
|
-
start?: number;
|
|
307
|
-
/** characters to keep at end, default 2 */
|
|
308
|
-
end?: number;
|
|
309
|
-
}) {
|
|
310
|
-
const mask = options && options.mask || "*****";
|
|
311
|
-
const start = options && isNumber(options.start) ? options.start : 2;
|
|
312
|
-
const end = options && isNumber(options.end) ? options.end : 2;
|
|
313
|
-
const prefix = start >= 0 ? str.slice(0, start) : str;
|
|
314
|
-
const sliceEnd = str.length - end;
|
|
315
|
-
const suffix = sliceEnd >= 0 ? str.slice(sliceEnd) : str;
|
|
316
|
-
|
|
317
|
-
return `${prefix}${mask}${suffix}`;
|
|
1
|
+
import { IDictionary } from "../types/common.types";
|
|
2
|
+
import { makeUniqueArray } from "./collections.base";
|
|
3
|
+
import { isNullOrEmptyString, isNullOrUndefined, isNumber, isString } from "./typecheckers";
|
|
4
|
+
|
|
5
|
+
export function endsWith(str: string, value: string, ignoreCase?: boolean): boolean {
|
|
6
|
+
let str1 = str;
|
|
7
|
+
let find = value;
|
|
8
|
+
if (ignoreCase) {
|
|
9
|
+
str1 = str1.toLowerCase();
|
|
10
|
+
find = find.toLowerCase();
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
return str1.substr(str1.length - find.length) === find;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export function startsWith(str: string, value: string, ignoreCase?: boolean): boolean {
|
|
17
|
+
let str1 = str;
|
|
18
|
+
let find = value;
|
|
19
|
+
if (ignoreCase) {
|
|
20
|
+
str1 = str1.toLowerCase();
|
|
21
|
+
find = find.toLowerCase();
|
|
22
|
+
}
|
|
23
|
+
return str1.substr(0, find.length) === find;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/** remove space at start or end */
|
|
27
|
+
export function trim(str: string): string {
|
|
28
|
+
return str.replace(/^\s+|\s+$/g, '');
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function trimEnd(str: string): string {
|
|
32
|
+
return str.replace(/\s+$/, "");
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export function trimStart(str: string): string {
|
|
36
|
+
return str.replace(/^\s+/, "");
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export function splice(str: string, start: number, delCount: number, insert: string) {
|
|
40
|
+
return str.slice(0, start) + insert + str.slice(start + Math.abs(delCount));
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export function padRight(value: string | number, length: number, fillString: string = "0") {
|
|
44
|
+
if (isNumber(value))
|
|
45
|
+
value = value.toString(10);
|
|
46
|
+
let pad = isNullOrEmptyString(fillString) ? "0" : fillString[0];
|
|
47
|
+
return value + Array(length - value.length + 1).join(pad);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export function padLeft(value: string | number, length: number, fillString: string = "0") {
|
|
51
|
+
if (isNumber(value))
|
|
52
|
+
value = value.toString(10);
|
|
53
|
+
let pad = isNullOrEmptyString(fillString) ? "0" : fillString[0];
|
|
54
|
+
return Array(length - String(value).length + 1).join(pad) + value;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/** returns array of [token] found inside the string
|
|
58
|
+
* supports token formats [Author??Created by {0}::Unknown author] will return "Author" as the token
|
|
59
|
+
* */
|
|
60
|
+
export function GetTokens(StringFormat: string): string[] {
|
|
61
|
+
let tokensResult: string[] = [];
|
|
62
|
+
if (isNullOrEmptyString(StringFormat)) return tokensResult;
|
|
63
|
+
|
|
64
|
+
let tokens = StringFormat.match(/\[[^\]]*\]/g);
|
|
65
|
+
if (tokens && tokens.length > 0) {
|
|
66
|
+
tokens.forEach(token => {
|
|
67
|
+
let key = token.slice(1, token.length - 1);
|
|
68
|
+
key = GetTokenInfo(key).tokenName;
|
|
69
|
+
if (tokensResult.indexOf(key) < 0)
|
|
70
|
+
tokensResult.push(key);
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
return tokensResult;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/** replaces a string with [token] and [otherToken] with their matched provided values
|
|
78
|
+
* supports token formats [Author??Created by {0}::Unknown author]
|
|
79
|
+
*/
|
|
80
|
+
export function ReplaceTokens(StringFormat: string, TokenValues: IDictionary<string>, options?: {
|
|
81
|
+
/**set to true if you want to keep "[token]" in the string when a token value wasn't provided */
|
|
82
|
+
keepMissingTokens?: boolean;
|
|
83
|
+
}): string {
|
|
84
|
+
|
|
85
|
+
let skipMissingTokens = options && options.keepMissingTokens;
|
|
86
|
+
|
|
87
|
+
if (isNullOrUndefined(StringFormat)) return null;
|
|
88
|
+
if (StringFormat !== '') {
|
|
89
|
+
let tokens = StringFormat.match(/\[[^\]]*\]/g);
|
|
90
|
+
if (tokens && tokens.length > 0) {
|
|
91
|
+
if (isNullOrUndefined(TokenValues)) TokenValues = {};
|
|
92
|
+
|
|
93
|
+
tokens.forEach(token => {
|
|
94
|
+
let key = token.slice(1, token.length - 1);
|
|
95
|
+
let tokenInfo = GetTokenInfo(key);
|
|
96
|
+
let value = TokenValues[tokenInfo.tokenName];
|
|
97
|
+
let skip = false;
|
|
98
|
+
if (isNullOrUndefined(value)) {
|
|
99
|
+
value = "";
|
|
100
|
+
skip = skipMissingTokens;//if true we won't replace this one
|
|
101
|
+
}
|
|
102
|
+
if (!skip || tokenInfo.hasFormat)
|
|
103
|
+
StringFormat = StringFormat.replace(token, tokenInfo.getValue(value));
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return StringFormat;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/** calls replace tokens on every dictionary value, or set null keys to "". This function ignores null/empty dictionaries */
|
|
111
|
+
export function ReplaceTokensInDictionary(Dictionary: IDictionary<string>, TokenValues: IDictionary<string>) {
|
|
112
|
+
if (!isNullOrUndefined(Dictionary)) {
|
|
113
|
+
Object.keys(Dictionary).forEach(key => {
|
|
114
|
+
Dictionary[key] = isNullOrEmptyString(Dictionary[key]) ? "" : ReplaceTokens(Dictionary[key], TokenValues);
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/** Normalizes a guid string, lower case and removes {} */
|
|
120
|
+
export function normalizeGuid(text: string, removeDashes?: boolean): string {
|
|
121
|
+
if (isNullOrEmptyString(text) || !isString(text)) {
|
|
122
|
+
return text;
|
|
123
|
+
}
|
|
124
|
+
var guid = text.toLowerCase().trim();
|
|
125
|
+
|
|
126
|
+
if (guid.startsWith("{")) {
|
|
127
|
+
guid = guid.substr(1);
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
if (guid.endsWith("}")) {
|
|
131
|
+
guid = guid.substr(0, guid.length - 1);
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
if (removeDashes) {
|
|
135
|
+
guid = guid.replace(/-/g, '');
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
return guid;
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
export function isEmptyGuid(guid: { toString(): string; }) {
|
|
142
|
+
if (isNullOrEmptyString(guid)) return true;
|
|
143
|
+
else if (Number(normalizeGuid(guid.toString(), true)) === 0) return true;
|
|
144
|
+
return false;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
export function escapeRegExp(text: string) {
|
|
148
|
+
return text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&');
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
export function isValidDomainLogin(login: string) {
|
|
152
|
+
return /^[A-Za-z0-9\\._-]{7,}$/.test(login);
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
export function stripRichTextWhitespace(value: string) {
|
|
156
|
+
// richText fields in have random markup even when field is empty
|
|
157
|
+
// \u200B zero width space
|
|
158
|
+
// \u200C zero width non-joiner Unicode code point
|
|
159
|
+
// \u200D zero width joiner Unicode code point
|
|
160
|
+
// \uFEFF zero width no-break space Unicode code point
|
|
161
|
+
return isString(value) ? value.replace(/[\u200B-\u200D\uFEFF]/g, "") : value;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
/** allows min length 1, letters, numbers underscore only */
|
|
165
|
+
export function isValidVarName(text: string) {
|
|
166
|
+
return /^[A-Za-z0-9_]{1,}$/.test(text);
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/** allows min length 1, letters, numbers underscore and hyphen only */
|
|
170
|
+
export function isValidHeaderName(text: string) {
|
|
171
|
+
return /^[A-Za-z0-9_-]{1,}$/.test(text);
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
/** returns token info with format */
|
|
175
|
+
export function GetTokenInfo(text: string) {
|
|
176
|
+
let split = text.split('??');
|
|
177
|
+
let hasFormat = split.length > 0 && !isNullOrEmptyString(split[1]);
|
|
178
|
+
let formatSplit = hasFormat ? split[1].split('::') : [];
|
|
179
|
+
let valueIfEmpty = formatSplit.length > 1 ? formatSplit[1] : "";
|
|
180
|
+
let formatIfNotEmpty = formatSplit[0];
|
|
181
|
+
let info = {
|
|
182
|
+
tokenName: hasFormat ? split[0] : text,
|
|
183
|
+
hasFormat: hasFormat,
|
|
184
|
+
getValue: (value: string) => {
|
|
185
|
+
if (!hasFormat) return value;
|
|
186
|
+
else {
|
|
187
|
+
if (isNullOrEmptyString(value)) return valueIfEmpty;
|
|
188
|
+
else return formatIfNotEmpty.replace('{0}', value);
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
};
|
|
192
|
+
return info;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
/** return true if both strings are the same, or both are empty/null/undefined */
|
|
196
|
+
export function stringEqualsOrEmpty(str1: string, str2: string, ignoreCase?: boolean) {
|
|
197
|
+
if (isNullOrEmptyString(str1) && isNullOrEmptyString(str2)) return true;
|
|
198
|
+
if (ignoreCase) {
|
|
199
|
+
if (!isNullOrEmptyString(str1)) str1 = str1.toLowerCase();
|
|
200
|
+
if (!isNullOrEmptyString(str2)) str2 = str2.toLowerCase();
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
return str1 === str2;
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
/** return true if str1 contains str2 */
|
|
207
|
+
export function stringContains(str1: string, str2: string, ignoreCase?: boolean) {
|
|
208
|
+
if (isNullOrEmptyString(str1) && isNullOrEmptyString(str2)) return true;
|
|
209
|
+
|
|
210
|
+
if (isNullOrEmptyString(str1))
|
|
211
|
+
str1 = "";
|
|
212
|
+
if (isNullOrEmptyString(str2))
|
|
213
|
+
str2 = "";
|
|
214
|
+
|
|
215
|
+
if (ignoreCase) {
|
|
216
|
+
str1 = str1.toLowerCase();
|
|
217
|
+
str2 = str2.toLowerCase();
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
return str1.indexOf(str2) >= 0;
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
export function cleanupString(str: string, options: {
|
|
224
|
+
replaceNewLines?: string;
|
|
225
|
+
collapseMultipleSpaces?: string;
|
|
226
|
+
collapseMultipleDashes?: string;
|
|
227
|
+
collapseMultipleUnderscore?: string;
|
|
228
|
+
}) {
|
|
229
|
+
if (isString(options.replaceNewLines))
|
|
230
|
+
str = str.replace(/\r/g, '')//no returns
|
|
231
|
+
.replace(/\n/g, options.replaceNewLines);//no line breaks
|
|
232
|
+
if (isString(options.collapseMultipleDashes))
|
|
233
|
+
str = str.replace(/-+/g, options.collapseMultipleSpaces);//no extra spaces
|
|
234
|
+
if (isString(options.collapseMultipleUnderscore))
|
|
235
|
+
str = str.replace(/_+/g, options.collapseMultipleUnderscore);//no extra spaces
|
|
236
|
+
|
|
237
|
+
// do this last, so it will collapse spaces added by previous options
|
|
238
|
+
if (isString(options.collapseMultipleSpaces)) {
|
|
239
|
+
str = str.replace(new RegExp(String.fromCharCode(160), "g"), '')//get rid of non-breaking spaces
|
|
240
|
+
.replace(/ +/g, options.collapseMultipleSpaces);//no extra spaces
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
return str;
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
/** normalizes   to see Issue 752 */
|
|
247
|
+
export function normalizeHtmlSpace(html: string) {
|
|
248
|
+
if (isNullOrEmptyString(html)) return html;
|
|
249
|
+
return html.replace(/ /i, " ");
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
export function replaceAll(str: string, find: string, replace: string, ignoreCase = false) {
|
|
253
|
+
//must call escapeRegExp on find, to make sure it works when there are protected regex characters
|
|
254
|
+
return str.replace(new RegExp(escapeRegExp(find), `g${ignoreCase ? 'i' : ''}`), replace);
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
export function capitalizeFirstLetter(str: string) {
|
|
258
|
+
return isNullOrEmptyString(str)
|
|
259
|
+
? ""
|
|
260
|
+
: `${str.charAt(0).toUpperCase()}${str.substring(1)}`;
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
export function escapeXml(unsafe: string, isAttribute = false) {
|
|
264
|
+
if (isNullOrEmptyString(unsafe)) return "";
|
|
265
|
+
return isAttribute
|
|
266
|
+
? unsafe.replace(/[<>&'"]/g, (c) => {
|
|
267
|
+
switch (c) {
|
|
268
|
+
case '<': return '<';
|
|
269
|
+
case '>': return '>';
|
|
270
|
+
case '&': return '&';
|
|
271
|
+
case '\'': return ''';
|
|
272
|
+
case '"': return '"';
|
|
273
|
+
}
|
|
274
|
+
return c;
|
|
275
|
+
})
|
|
276
|
+
: unsafe.replace(/[<>&]/g, (c) => {
|
|
277
|
+
switch (c) {
|
|
278
|
+
case '<': return '<';
|
|
279
|
+
case '>': return '>';
|
|
280
|
+
case '&': return '&';
|
|
281
|
+
}
|
|
282
|
+
return c;
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
/** uses regex str.match to replace each match by calling the replacer function (imported from CMS) */
|
|
287
|
+
export function replaceRegex(str: string, regex: RegExp, replacer: (match: string) => string | null) {
|
|
288
|
+
let matches = str.match(regex);
|
|
289
|
+
if (!matches || matches.length < 1) return str;
|
|
290
|
+
//replace each found token only once
|
|
291
|
+
let unique = makeUniqueArray(matches);
|
|
292
|
+
//todo: this has a bug where tokens matched contain each other, example, match numbers and prefix with #: 10, 100 will match both and produce #10 ##100
|
|
293
|
+
unique.forEach(m => {
|
|
294
|
+
let replacement = replacer(m);
|
|
295
|
+
if (!isNullOrUndefined(replacement))//ignore nulls
|
|
296
|
+
str = replaceAll(str, m, replacement);
|
|
297
|
+
});
|
|
298
|
+
return str;
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
/** masks a long string, keeping X number for characters at the start/end and replacing the middle with the mask string (default: CC*****CCC) */
|
|
302
|
+
export function maskString(str: string, options?: {
|
|
303
|
+
/** mask string, default ***** */
|
|
304
|
+
mask?: string;
|
|
305
|
+
/** characters to keep at start, default 2 */
|
|
306
|
+
start?: number;
|
|
307
|
+
/** characters to keep at end, default 2 */
|
|
308
|
+
end?: number;
|
|
309
|
+
}) {
|
|
310
|
+
const mask = options && options.mask || "*****";
|
|
311
|
+
const start = options && isNumber(options.start) ? options.start : 2;
|
|
312
|
+
const end = options && isNumber(options.end) ? options.end : 2;
|
|
313
|
+
const prefix = start >= 0 ? str.slice(0, start) : str;
|
|
314
|
+
const sliceEnd = str.length - end;
|
|
315
|
+
const suffix = sliceEnd >= 0 ? str.slice(sliceEnd) : str;
|
|
316
|
+
|
|
317
|
+
return `${prefix}${mask}${suffix}`;
|
|
318
318
|
}
|