wisp
Version:
Homoiconic JS with clojure syntax, s-expressions & macros
129 lines (128 loc) • 20.5 kB
JavaScript
{
var _ns_ = {
id: 'wisp.string',
doc: void 0
};
var wisp_runtime = require('./runtime');
var isFn = wisp_runtime.isFn;
var str = wisp_runtime.str;
var subs = wisp_runtime.subs;
var reMatches = wisp_runtime.reMatches;
var isNil = wisp_runtime.isNil;
var isString = wisp_runtime.isString;
var isRePattern = wisp_runtime.isRePattern;
var dec = wisp_runtime.dec;
var max = wisp_runtime.max;
var wisp_sequence = require('./sequence');
var seq = wisp_sequence.seq;
var lazySeq = wisp_sequence.lazySeq;
var vec = wisp_sequence.vec;
var conj = wisp_sequence.conj;
var cons = wisp_sequence.cons;
var first = wisp_sequence.first;
var rest = wisp_sequence.rest;
var take = wisp_sequence.take;
var count = wisp_sequence.count;
var isEmpty = wisp_sequence.isEmpty;
}
var reFindAll = exports.reFindAll = isFn(''.matchAll) ? function reFindAll(re, s) {
return seq(s.matchAll(RegExp(re, 'g')));
} : function reFindAll(re, s) {
return function rec(suffix, prefix) {
return function () {
var xø1 = suffix.match(re);
return xø1 ? function () {
var posø1 = xø1.index + max(1, count(first(xø1)));
Object.assign(xø1, {
'input': s,
'index': prefix + xø1.index
});
return isEmpty(suffix) ? lazySeq.call(void 0, false, function () {
return [xø1];
}) : lazySeq.call(void 0, false, function () {
return cons(xø1, rec(subs(suffix, posø1), prefix + posø1));
});
}.call(this) : void 0;
}.call(this);
}(s, 0);
};
var clojureSplit = function clojureSplit(string, pattern, limit) {
return function loop() {
var recur = loop;
var matchesø1 = take(dec(limit), reFindAll(pattern, string));
var resø1 = [];
var indexø1 = 0;
do {
recur = isEmpty(matchesø1) ? conj(resø1, subs(string, indexø1)) : function () {
var xø1 = first(matchesø1);
return loop[0] = rest(matchesø1), loop[1] = conj(resø1, subs(string, indexø1, xø1.index)), loop[2] = xø1.index + count(first(xø1)), loop;
}.call(this);
} while (matchesø1 = loop[0], resø1 = loop[1], indexø1 = loop[2], recur === loop);
return recur;
}.call(this);
};
var split = exports.split = function split(string, pattern, limit) {
return !limit ? string.split(pattern) : clojureSplit(string, pattern, limit > 0 ? limit : Infinity);
};
var splitLines = exports.splitLines = function splitLines(s) {
return split(s, /\n|\r\n/);
};
var join = exports.join = function join() {
switch (arguments.length) {
case 1:
var coll = arguments[0];
return str.apply(void 0, vec(coll));
case 2:
var separator = arguments[0];
var coll = arguments[1];
return vec(coll).join(separator);
default:
throw RangeError('Wrong number of arguments passed');
}
};
var upperCase = exports.upperCase = function upperCase(string) {
return string.toUpperCase();
};
var lowerCase = exports.lowerCase = function lowerCase(string) {
return string.toLowerCase();
};
var capitalize = exports.capitalize = function capitalize(s) {
return count(s) < 2 ? upperCase(s) : '' + upperCase(subs(s, 0, 1)) + lowerCase(subs(s, 1));
};
var ESCAPE_PATTERN = new RegExp('([-()\\[\\]{}+?*.$\\^|,:#<!\\\\])', 'g');
var patternEscape = exports.patternEscape = function patternEscape(source) {
return source.replace(ESCAPE_PATTERN, '\\$1').replace(new RegExp('\\x08', 'g'), '\\x08');
};
var replaceFirst = exports.replaceFirst = function replaceFirst(string, match, replacement) {
return string.replace(match, replacement);
};
var replace = exports.replace = function replace(string, match, replacement) {
return isString(match) ? string.replace(new RegExp(patternEscape(match), 'g'), replacement) : isRePattern(match) ? string.replace(new RegExp(match.source, 'g'), replacement) : 'else' ? (function () {
throw '' + 'Invalid match arg: ' + match;
})() : void 0;
};
var __LEFTSPACES__ = exports.__LEFTSPACES__ = /^\s\s*/;
var __RIGHTSPACES__ = exports.__RIGHTSPACES__ = /\s\s*$/;
var __SPACES__ = exports.__SPACES__ = /^\s\s*$/;
var triml = exports.triml = isNil(''.trimLeft) ? function (string) {
return string.replace(__LEFTSPACES__, '');
} : function (string) {
return string.trimLeft();
};
var trimr = exports.trimr = isNil(''.trimRight) ? function (string) {
return string.replace(__RIGHTSPACES__, '');
} : function (string) {
return string.trimRight();
};
var trim = exports.trim = isNil(''.trim) ? function (string) {
return string.replace(__LEFTSPACES__).replace(__RIGHTSPACES__);
} : function (string) {
return string.trim();
};
var isBlank = exports.isBlank = function isBlank(string) {
return isNil(string) || isEmpty(string) || reMatches(__SPACES__, string);
};
var reverse = exports.reverse = function reverse(string) {
return join('', string.split(/(?:)/).reverse());
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["anonymous.wisp"],"names":["_ns_","id","doc","isFn","str","subs","reMatches","isNil","isString","isRePattern","dec","max","seq","lazySeq","vec","conj","cons","first","rest","take","count","isEmpty","reFindAll","exports","matchAll","re","s","RegExp","rec","suffix","prefix","xø1","match","posø1","index","Object","assign","clojureSplit","string","pattern","limit","matchesø1","resø1","indexø1","split","Infinity","splitLines","join","coll","separator","upperCase","toUpperCase","lowerCase","toLowerCase","capitalize","ESCAPE_PATTERN","patternEscape","source","replace","replaceFirst","replacement","__LEFTSPACES__","__RIGHTSPACES__","__SPACES__","triml","trimLeft","trimr","trimRight","trim","isBlank","reverse"],"mappings":"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","sourcesContent":["(ns wisp.string\n  (:require [wisp.runtime :refer [fn? str subs re-matches nil? string? re-pattern? dec max]]\n            [wisp.sequence :refer [seq lazy-seq vec conj cons first rest take count empty?]]))\n\n(def\n  ^{:doc \"Returns all matches of pattern occurring in string (as is)\"}\n  re-find-all\n  (if (fn? (.-match-all \"\"))               ; Chrome 73+, Firefox 67+, Node 12+\n    (fn re-find-all [re s]\n      (seq (.match-all s (RegExp re \\g))))\n    (fn re-find-all [re s]\n      ((fn rec [suffix prefix]             ; simulating match-all behaviour\n         (let [x (.match suffix re)]\n           (if x\n             (let [pos (+ (.-index x) (max 1 (count (first x))))]\n               (Object.assign x {:input s, :index (+ prefix (.-index x))})\n               (if (empty? suffix)\n                 (lazy-seq [x])\n                 (lazy-seq (cons x (rec (subs suffix pos) (+ prefix pos)))))))))\n       s #_\"removing prefix to prevent repeat matches\"\n       0 #_\"keeping track of removed prefix length\"))))\n\n(defn- clojure-split [string pattern limit]\n  (loop [matches (take (dec limit) (re-find-all pattern string)),  res [],  index 0]\n    (if (empty? matches)\n      (conj res (subs string index))\n      (let [x (first matches)]\n        (recur (rest matches)\n               (conj res (subs string index (.-index x)))\n               (+ (.-index x) (count (first x))))))))\n\n(defn split\n  \"Splits string on a regular expression.  Optional argument limit is\n  the maximum number of splits. Not lazy. Returns vector of the splits.\"\n  [string pattern limit]\n  (if (not limit)\n    (.split string pattern)\n    (clojure-split string pattern (if (> limit 0) limit Infinity))))\n\n(defn split-lines\n  \"Splits s on \\n or \\r\\n.\"\n  [s]\n  (split s #\"\\n|\\r\\n\"))\n\n(defn join\n  \"Returns a string of all elements in coll, as returned by (seq coll),\n   separated by an optional separator.\"\n  ([coll]\n     (apply str (vec coll)))\n  ([separator coll]\n     (.join (vec coll) separator)))\n\n(defn upper-case\n  \"Converts string to all upper-case.\"\n  [string]\n  (.toUpperCase string))\n\n(defn lower-case\n  \"Converts string to all lower-case.\"\n  [string]\n  (.toLowerCase string))\n\n(defn ^String capitalize\n  \"Converts first character of the string to upper-case, all other\n  characters to lower-case.\"\n  [s]\n  (if (< (count s) 2)\n      (upper-case s)\n      (str (upper-case (subs s 0 1))\n           (lower-case (subs s 1)))))\n\n(def ^:private ESCAPE_PATTERN\n  (RegExp. \"([-()\\\\[\\\\]{}+?*.$\\\\^|,:#<!\\\\\\\\])\" \"g\"))\n\n(defn pattern-escape\n  [source]\n  (.replace (.replace source ESCAPE_PATTERN \"\\\\$1\")\n            (RegExp. \"\\\\x08\" \"g\"), \"\\\\x08\"))\n\n(defn replace-first\n  \"Replaces the first instance of match with replacement in s.\n  match/replacement can be:\n\n  string / string\n  pattern / (string or function of match).\"\n  [string match replacement]\n  (.replace string match replacement))\n\n(defn replace\n  \"Replaces all instance of match with replacement in s.\n\n   match/replacement can be:\n\n   string / string\n   char / char\n   pattern / (string or function of match).\n\n   See also replace-first.\"\n  [string match replacement]\n  (cond (string? match)\n        (.replace string (RegExp. (pattern-escape match) \"g\") replacement)\n\n        (re-pattern? match)\n        (.replace string (RegExp. (.-source match) \"g\") replacement)\n\n        :else\n        (throw (str \"Invalid match arg: \" match))))\n\n\n;(def **WHITESPACE** (str \"[\\x09\\x0A\\x0B\\x0C\\x0D\\x20\\xA0\\u1680\\u180E\\u2000\"\n;                          \"\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\"\n;                          \"\\u2009\\u200A\\u202F\\u205F\\u3000\\u2028\\u2029\\uFEFF]\"))\n;(def **LEFT-SPACES** (re-pattern (str \"^\" **WHITESPACE** **WHITESPACE** \"*\")))\n;(def **RIGHT-SPACES** (re-pattern (str **WHITESPACE** **WHITESPACE** \"*$\")))\n;(def **SPACES** (re-pattern (str \"^\" **WHITESPACE** \"*$\")))\n\n\n(def **LEFT-SPACES** #\"^\\s\\s*\")\n(def **RIGHT-SPACES** #\"\\s\\s*$\")\n(def **SPACES** #\"^\\s\\s*$\")\n\n\n(def\n  ^{:tag string\n    :doc \"Removes whitespace from the left side of string.\"}\n  triml\n  (if (nil? (.-trimLeft \"\"))\n    (fn [string] (.replace string **LEFT-SPACES** \"\"))\n    (fn [string] (.trimLeft string))))\n\n(def\n  ^{:tag string\n    :doc \"Removes whitespace from the right side of string.\"}\n  trimr\n  (if (nil? (.-trimRight \"\"))\n    (fn [string] (.replace string **RIGHT-SPACES** \"\"))\n    (fn [string] (.trimRight string))))\n\n(def\n  ^{:tag string\n    :doc \"Removes whitespace from both ends of string.\"}\n  trim\n  (if (nil? (.-trim \"\"))\n    (fn [string] (.replace (.replace string **LEFT-SPACES**) **RIGHT-SPACES**))\n    (fn [string] (.trim string))))\n\n(defn blank?\n  \"True if s is nil, empty, or contains only whitespace.\"\n  [string]\n  (or (nil? string)\n      (empty? string)\n      (re-matches **SPACES** string)))\n\n(defn reverse\n  \"Returns s with its characters reversed.\"\n  [string]\n  (join \"\" (.reverse (.split string #\"\"))))\n"]}