Mercurial > pub > ImplabJs
comparison src/implab/safe.js @ 0:fc2517695ee1
Initial commit, draft import of existing work
| author | cin |
|---|---|
| date | Thu, 01 Jun 2017 13:20:03 +0300 |
| parents | |
| children | 00779cb63b12 |
comparison
equal
deleted
inserted
replaced
| -1:000000000000 | 0:fc2517695ee1 |
|---|---|
| 1 define([], | |
| 2 | |
| 3 function () { | |
| 4 var id = 0, | |
| 5 OID_FIELD = "__core_safe_oid_field", | |
| 6 _create, _defineProperty, _keys; | |
| 7 | |
| 8 if (!Object.keys) { | |
| 9 _defineProperty = function (obj, prop, d) { | |
| 10 if (!d) | |
| 11 throw new Error("Invalid descriptor"); | |
| 12 obj[prop] = d.value; | |
| 13 }; | |
| 14 } else { | |
| 15 _defineProperty = Object.defineProperty; | |
| 16 } | |
| 17 | |
| 18 if (!Object.create) { | |
| 19 var tctor = function () {}; | |
| 20 _create = function (proto) { | |
| 21 if (arguments.length > 1) | |
| 22 throw new Error("The two arguments for isn't supported"); | |
| 23 | |
| 24 tctor.prototype = proto; | |
| 25 return new tctor(); | |
| 26 }; | |
| 27 } else { | |
| 28 _create = Object.create; | |
| 29 } | |
| 30 | |
| 31 if (!Object.keys) { | |
| 32 // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/keys#Polyfill | |
| 33 _keys = (function () { | |
| 34 'use strict'; | |
| 35 var hasOwnProperty = Object.prototype.hasOwnProperty, | |
| 36 hasDontEnumBug = !({ | |
| 37 toString: null | |
| 38 }).propertyIsEnumerable('toString'), | |
| 39 dontEnums = [ | |
| 40 'toString', | |
| 41 'toLocaleString', | |
| 42 'valueOf', | |
| 43 'hasOwnProperty', | |
| 44 'isPrototypeOf', | |
| 45 'propertyIsEnumerable', | |
| 46 'constructor' | |
| 47 ], | |
| 48 dontEnumsLength = dontEnums.length; | |
| 49 | |
| 50 return function (obj) { | |
| 51 if (typeof obj !== 'object' && (typeof obj !== 'function' || obj === null)) { | |
| 52 throw new TypeError('Object.keys called on non-object'); | |
| 53 } | |
| 54 | |
| 55 var result = [], | |
| 56 prop, i; | |
| 57 | |
| 58 for (prop in obj) { | |
| 59 if (prop != OID_FIELD && hasOwnProperty.call(obj, prop)) { | |
| 60 result.push(prop); | |
| 61 } | |
| 62 } | |
| 63 | |
| 64 if (hasDontEnumBug) { | |
| 65 for (i = 0; i < dontEnumsLength; i++) { | |
| 66 if (hasOwnProperty.call(obj, dontEnums[i])) { | |
| 67 result.push(dontEnums[i]); | |
| 68 } | |
| 69 } | |
| 70 } | |
| 71 return result; | |
| 72 }; | |
| 73 }()); | |
| 74 } else { | |
| 75 _keys = Object.keys; | |
| 76 } | |
| 77 | |
| 78 var safe = null; | |
| 79 safe = { | |
| 80 OID_FIELD: OID_FIELD, | |
| 81 | |
| 82 argumentNotNull: function (arg, name) { | |
| 83 if (arg === null || arg === undefined) | |
| 84 throw new Error("The argument " + name + " can't be null or undefined"); | |
| 85 }, | |
| 86 | |
| 87 argumentNotEmptyString: function (arg, name) { | |
| 88 if (typeof (arg) !== "string" || !arg.length) | |
| 89 throw new Error("The argument '" + name + "' must be a not empty string"); | |
| 90 }, | |
| 91 | |
| 92 argumentNotEmptyArray: function (arg, name) { | |
| 93 if (!(arg instanceof Array) || !arg.length) | |
| 94 throw new Error("The argument '" + name + "' must be a not empty array"); | |
| 95 }, | |
| 96 | |
| 97 argumentOfType: function (arg, type, name) { | |
| 98 if (!(arg instanceof type)) | |
| 99 throw new Error("The argument '" + name + "' type doesn't match"); | |
| 100 }, | |
| 101 | |
| 102 isNull: function (arg) { | |
| 103 return (arg === null || arg === undefined); | |
| 104 }, | |
| 105 | |
| 106 isPrimitive: function (arg) { | |
| 107 return (arg === null || arg === undefined || typeof (arg) === "string" || | |
| 108 typeof (arg) === "number" || typeof (arg) === "boolean"); | |
| 109 }, | |
| 110 | |
| 111 isInteger: function (arg) { | |
| 112 return parseInt(arg) === arg; | |
| 113 }, | |
| 114 | |
| 115 isNumber: function (arg) { | |
| 116 return parseFloat(arg) === arg; | |
| 117 }, | |
| 118 | |
| 119 isString: function (val) { | |
| 120 return typeof (val) == "string" || val instanceof String; | |
| 121 }, | |
| 122 | |
| 123 isNullOrEmptyString: function (str) { | |
| 124 if (str === null || str === undefined || | |
| 125 ((typeof (str) == "string" || str instanceof String) && str.length === 0)) | |
| 126 return true; | |
| 127 }, | |
| 128 | |
| 129 isNotEmptyArray: function (arg) { | |
| 130 return (arg instanceof Array && arg.length > 0); | |
| 131 }, | |
| 132 | |
| 133 /** | |
| 134 * Выполняет метод для каждого элемента массива, останавливается, когда | |
| 135 * либо достигнут конец массива, либо функция <c>cb</c> вернула | |
| 136 * значение. | |
| 137 * | |
| 138 * @param{Array | Object} obj массив элементов для просмотра | |
| 139 * @param{Function} cb функция, вызываемая для каждого элемента | |
| 140 * @param{Object} thisArg значение, которое будет передано в качестве | |
| 141 * <c>this</c> в <c>cb</c>. | |
| 142 * @returns Результат вызова функции <c>cb</c>, либо <c>undefined</c> | |
| 143 * если достигнут конец массива. | |
| 144 */ | |
| 145 each: function (obj, cb, thisArg) { | |
| 146 safe.argumentNotNull(cb, "cb"); | |
| 147 var i, x; | |
| 148 if (obj instanceof Array) { | |
| 149 for (i = 0; i < obj.length; i++) { | |
| 150 x = cb.call(thisArg, obj[i], i); | |
| 151 if (x !== undefined) | |
| 152 return x; | |
| 153 } | |
| 154 } else { | |
| 155 var keys = _keys(obj); | |
| 156 for (i = 0; i < keys.length; i++) { | |
| 157 var k = keys[i]; | |
| 158 x = cb.call(thisArg, obj[k], k); | |
| 159 if (x !== undefined) | |
| 160 return x; | |
| 161 } | |
| 162 } | |
| 163 }, | |
| 164 | |
| 165 oid: function (obj) { | |
| 166 return this.isPrimitive(obj) ? undefined : obj[OID_FIELD] || | |
| 167 (_defineProperty(obj, OID_FIELD, { | |
| 168 value: ++id, | |
| 169 enumerable: false | |
| 170 }) && id); | |
| 171 }, | |
| 172 | |
| 173 /** | |
| 174 * Копирует свойства одного объекта в другой. | |
| 175 * | |
| 176 * @param{Any} dest объект в который нужно скопировать значения | |
| 177 * @param{Any} src источник из которого будут копироваться значения | |
| 178 * @tmpl{Object|Array} tmpl шаблон по которому будет происходить | |
| 179 * копирование. Если шаблон является массивом | |
| 180 * (список свойств), тогда значения этого массива | |
| 181 * являются именами свойсвт которые будут | |
| 182 * скопированы. Если шаблон является объектом (карта | |
| 183 * преобразования имен свойств src->dst), тогда | |
| 184 * копирование будет осуществляться только | |
| 185 * собственных свойств источника, присутсвующих в | |
| 186 * шаблоне, при этом значение свойства шаблона | |
| 187 * является именем свойства в которое будет | |
| 188 * произведено коприрование | |
| 189 */ | |
| 190 mixin: function (dest, src, tmpl) { | |
| 191 safe.argumentNotNull(dest, "dest"); | |
| 192 if (!src) | |
| 193 return dest; | |
| 194 | |
| 195 var keys, i, p; | |
| 196 if (arguments.length < 3) { | |
| 197 keys = _keys(src); | |
| 198 for (i = 0; i < keys.length; i++) { | |
| 199 p = keys[i]; | |
| 200 dest[p] = src[p]; | |
| 201 } | |
| 202 } else { | |
| 203 if (tmpl instanceof Array) { | |
| 204 for (i = 0; i < tmpl.length; i++) { | |
| 205 p = tmpl[i]; | |
| 206 if (p in src) | |
| 207 dest[p] = src[p]; | |
| 208 } | |
| 209 | |
| 210 } else { | |
| 211 keys = _keys(src); | |
| 212 for (i = 0; i < keys.length; i++) { | |
| 213 p = keys[i]; | |
| 214 if (p in tmpl) | |
| 215 dest[tmpl[p]] = src[p]; | |
| 216 } | |
| 217 } | |
| 218 } | |
| 219 return dest; | |
| 220 }, | |
| 221 | |
| 222 create: _create, | |
| 223 | |
| 224 delegate: function (target, method) { | |
| 225 if (!(method instanceof Function)) { | |
| 226 this.argumentNotNull(target, "target"); | |
| 227 method = target[method]; | |
| 228 } | |
| 229 | |
| 230 if (!(method instanceof Function)) | |
| 231 throw new Error("'method' argument must be a Function or a method name"); | |
| 232 | |
| 233 return function () { | |
| 234 return method.apply(target, arguments); | |
| 235 }; | |
| 236 }, | |
| 237 | |
| 238 /** | |
| 239 * Для каждого элемента массива вызывает указанную функцию и сохраняет | |
| 240 * возвращенное значение в массиве результатов. | |
| 241 * | |
| 242 * @remarks cb может выполняться асинхронно, при этом одновременно будет | |
| 243 * только одна операция. | |
| 244 * | |
| 245 * @async | |
| 246 */ | |
| 247 pmap: function (items, cb) { | |
| 248 safe.argumentNotNull(cb, "cb"); | |
| 249 | |
| 250 if (items && items.then instanceof Function) | |
| 251 return items.then(function (data) { | |
| 252 return safe.pmap(data, cb); | |
| 253 }); | |
| 254 | |
| 255 if (safe.isNull(items) || !items.length) | |
| 256 return items; | |
| 257 | |
| 258 var i = 0, | |
| 259 result = []; | |
| 260 | |
| 261 function next() { | |
| 262 var r, ri; | |
| 263 | |
| 264 function chain(x) { | |
| 265 result[ri] = x; | |
| 266 return next(); | |
| 267 } | |
| 268 | |
| 269 while (i < items.length) { | |
| 270 r = cb(items[i], i); | |
| 271 ri = i; | |
| 272 i++; | |
| 273 if (r && r.then) { | |
| 274 return r.then(chain); | |
| 275 } else { | |
| 276 result[ri] = r; | |
| 277 } | |
| 278 } | |
| 279 return result; | |
| 280 } | |
| 281 | |
| 282 return next(); | |
| 283 }, | |
| 284 | |
| 285 /** | |
| 286 * Для каждого элемента массива вызывает указанную функцию, результаты | |
| 287 * не сохраняются | |
| 288 * | |
| 289 * @remarks cb может выполняться асинхронно, при этом одновременно будет | |
| 290 * только одна операция. | |
| 291 * @async | |
| 292 */ | |
| 293 pfor: function (items, cb) { | |
| 294 safe.argumentNotNull(cb, "cb"); | |
| 295 | |
| 296 if (items && items.then instanceof Function) | |
| 297 return items.then(function (data) { | |
| 298 return safe.pmap(data, cb); | |
| 299 }); | |
| 300 | |
| 301 if (safe.isNull(items) || !items.length) | |
| 302 return items; | |
| 303 | |
| 304 var i = 0; | |
| 305 | |
| 306 function next() { | |
| 307 while (i < items.length) { | |
| 308 var r = cb(items[i], i); | |
| 309 i++; | |
| 310 if (r && r.then) | |
| 311 return r.then(next); | |
| 312 } | |
| 313 } | |
| 314 | |
| 315 return next(); | |
| 316 }, | |
| 317 | |
| 318 /** | |
| 319 * Выбирает первый элемент из последовательности, или обещания, если в | |
| 320 * качестве параметра используется обещание, оно должно вернуть массив. | |
| 321 * | |
| 322 * @param{Function} cb обработчик результата, ему будет передан первый | |
| 323 * элемент последовательности в случае успеха | |
| 324 * @param{Fucntion} err обработчик исключения, если массив пустой, либо | |
| 325 * не массив | |
| 326 * | |
| 327 * @remarks Если не указаны ни cb ни err, тогда функция вернет либо | |
| 328 * обещание, либо первый элемент. | |
| 329 * @async | |
| 330 */ | |
| 331 first: function (sequence, cb, err) { | |
| 332 if (sequence) { | |
| 333 if (sequence.then instanceof Function) { | |
| 334 return sequence.then(function (res) { | |
| 335 return safe.first(res, cb, err); | |
| 336 }, err); | |
| 337 } else if (sequence && "length" in sequence) { | |
| 338 if (sequence.length === 0) { | |
| 339 if (err) | |
| 340 return err(new Error("The sequence is empty")); | |
| 341 else | |
| 342 throw new Error("The sequence is empty"); | |
| 343 } | |
| 344 return cb ? cb(sequence[0]) : sequence[0]; | |
| 345 } | |
| 346 } | |
| 347 | |
| 348 if (err) | |
| 349 return err(new Error("The sequence is required")); | |
| 350 else | |
| 351 throw new Error("The sequence is required"); | |
| 352 } | |
| 353 }; | |
| 354 | |
| 355 return safe; | |
| 356 }); |
