1 | // Copyright (C) 2016 The Qt Company Ltd. |
---|---|
2 | // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only |
3 | |
4 | |
5 | #include "qv4stringobject_p.h" |
6 | #include "qv4regexp_p.h" |
7 | #include "qv4regexpobject_p.h" |
8 | #include <private/qv4mm_p.h> |
9 | #include "qv4scopedvalue_p.h" |
10 | #include "qv4symbol_p.h" |
11 | #include <private/qv4alloca_p.h> |
12 | #include "qv4jscall_p.h" |
13 | #include "qv4stringiterator_p.h" |
14 | #include <QtCore/QDateTime> |
15 | #include <QtCore/QDebug> |
16 | #include <QtCore/QStringList> |
17 | #include <QtQml/private/qv4runtime_p.h> |
18 | |
19 | #include <cassert> |
20 | |
21 | #ifndef Q_OS_WIN |
22 | # include <time.h> |
23 | # ifndef Q_OS_VXWORKS |
24 | # include <sys/time.h> |
25 | # else |
26 | # include "qplatformdefs.h" |
27 | # endif |
28 | #else |
29 | # include <qt_windows.h> |
30 | #endif |
31 | |
32 | using namespace QV4; |
33 | using namespace Qt::Literals::StringLiterals; |
34 | |
35 | DEFINE_OBJECT_VTABLE(StringObject); |
36 | |
37 | void Heap::StringObject::init() |
38 | { |
39 | Object::init(); |
40 | Q_ASSERT(vtable() == QV4::StringObject::staticVTable()); |
41 | string.set(e: internalClass->engine, newVal: internalClass->engine->id_empty()->d()); |
42 | setProperty(e: internalClass->engine, index: LengthPropertyIndex, v: Value::fromInt32(i: 0)); |
43 | } |
44 | |
45 | void Heap::StringObject::init(const QV4::String *str) |
46 | { |
47 | Object::init(); |
48 | string.set(e: internalClass->engine, newVal: str->d()); |
49 | setProperty(e: internalClass->engine, index: LengthPropertyIndex, v: Value::fromInt32(i: length())); |
50 | } |
51 | |
52 | Heap::String *Heap::StringObject::getIndex(uint index) const |
53 | { |
54 | QString str = string->toQString(); |
55 | if (index >= (uint)str.size()) |
56 | return nullptr; |
57 | return internalClass->engine->newString(s: str.mid(position: index, n: 1)); |
58 | } |
59 | |
60 | uint Heap::StringObject::length() const |
61 | { |
62 | return string->length(); |
63 | } |
64 | |
65 | bool StringObject::virtualDeleteProperty(Managed *m, PropertyKey id) |
66 | { |
67 | Q_ASSERT(m->as<StringObject>()); |
68 | if (id.isArrayIndex()) { |
69 | StringObject *o = static_cast<StringObject *>(m); |
70 | uint index = id.asArrayIndex(); |
71 | if (index < static_cast<uint>(o->d()->string->toQString().size())) |
72 | return false; |
73 | } |
74 | return Object::virtualDeleteProperty(m, id); |
75 | } |
76 | |
77 | struct StringObjectOwnPropertyKeyIterator : ObjectOwnPropertyKeyIterator |
78 | { |
79 | ~StringObjectOwnPropertyKeyIterator() override = default; |
80 | PropertyKey next(const QV4::Object *o, Property *pd = nullptr, PropertyAttributes *attrs = nullptr) override; |
81 | |
82 | }; |
83 | |
84 | PropertyKey StringObjectOwnPropertyKeyIterator::next(const QV4::Object *o, Property *pd, PropertyAttributes *attrs) |
85 | { |
86 | const StringObject *s = static_cast<const StringObject *>(o); |
87 | uint slen = s->d()->string->toQString().size(); |
88 | if (arrayIndex < slen) { |
89 | uint index = arrayIndex; |
90 | ++arrayIndex; |
91 | if (attrs) |
92 | *attrs = Attr_NotConfigurable|Attr_NotWritable; |
93 | if (pd) |
94 | pd->value = s->getIndex(index); |
95 | return PropertyKey::fromArrayIndex(idx: index); |
96 | } else if (arrayIndex == slen) { |
97 | if (s->arrayData()) { |
98 | SparseArrayNode *arrayNode = s->sparseBegin(); |
99 | // iterate until we're past the end of the string |
100 | while (arrayNode && arrayNode->key() < slen) |
101 | arrayNode = arrayNode->nextNode(); |
102 | } |
103 | } |
104 | |
105 | return ObjectOwnPropertyKeyIterator::next(o, pd, attrs); |
106 | } |
107 | |
108 | OwnPropertyKeyIterator *StringObject::virtualOwnPropertyKeys(const Object *m, Value *target) |
109 | { |
110 | *target = *m; |
111 | return new StringObjectOwnPropertyKeyIterator; |
112 | } |
113 | |
114 | PropertyAttributes StringObject::virtualGetOwnProperty(const Managed *m, PropertyKey id, Property *p) |
115 | { |
116 | PropertyAttributes attributes = Object::virtualGetOwnProperty(m, id, p); |
117 | if (attributes != Attr_Invalid) |
118 | return attributes; |
119 | |
120 | if (id.isArrayIndex()) { |
121 | const uint index = id.asArrayIndex(); |
122 | const auto s = static_cast<const StringObject *>(m); |
123 | if (index < uint(s->d()->string->toQString().size())) { |
124 | if (p) |
125 | p->value = s->getIndex(index); |
126 | return Attr_NotConfigurable|Attr_NotWritable; |
127 | } |
128 | } |
129 | return Object::virtualGetOwnProperty(m, id, p); |
130 | } |
131 | |
132 | DEFINE_OBJECT_VTABLE(StringCtor); |
133 | |
134 | void Heap::StringCtor::init(QV4::ExecutionEngine *engine) |
135 | { |
136 | Heap::FunctionObject::init(engine, QStringLiteral("String")); |
137 | } |
138 | |
139 | ReturnedValue StringCtor::virtualCallAsConstructor(const FunctionObject *f, const Value *argv, int argc, const Value *newTarget) |
140 | { |
141 | ExecutionEngine *v4 = static_cast<const Object *>(f)->engine(); |
142 | Scope scope(v4); |
143 | ScopedString value(scope); |
144 | if (argc) |
145 | value = argv[0].toString(e: v4); |
146 | else |
147 | value = v4->newString(); |
148 | CHECK_EXCEPTION(); |
149 | ReturnedValue o = Encode(v4->newStringObject(string: value)); |
150 | |
151 | if (!newTarget) |
152 | return o; |
153 | ScopedObject obj(scope, o); |
154 | obj->setProtoFromNewTarget(newTarget); |
155 | return obj->asReturnedValue(); |
156 | } |
157 | |
158 | ReturnedValue StringCtor::virtualCall(const FunctionObject *m, const Value *, const Value *argv, int argc) |
159 | { |
160 | ExecutionEngine *v4 = m->engine(); |
161 | if (!argc) |
162 | return v4->newString()->asReturnedValue(); |
163 | if (argv[0].isSymbol()) |
164 | return v4->newString(s: argv[0].symbolValue()->descriptiveString())->asReturnedValue(); |
165 | return argv[0].toString(e: v4)->asReturnedValue(); |
166 | } |
167 | |
168 | ReturnedValue StringCtor::method_fromCharCode(const FunctionObject *b, const Value *, const Value *argv, int argc) |
169 | { |
170 | QString str(argc, Qt::Uninitialized); |
171 | QChar *ch = str.data(); |
172 | for (int i = 0, ei = argc; i < ei; ++i) { |
173 | *ch = QChar(argv[i].toUInt16()); |
174 | ++ch; |
175 | } |
176 | return Encode(b->engine()->newString(s: str)); |
177 | } |
178 | |
179 | |
180 | |
181 | ReturnedValue StringCtor::method_fromCodePoint(const FunctionObject *f, const Value *, const Value *argv, int argc) |
182 | { |
183 | ExecutionEngine *e = f->engine(); |
184 | QString result(argc*2, Qt::Uninitialized); // assume worst case |
185 | QChar *ch = result.data(); |
186 | for (int i = 0; i < argc; ++i) { |
187 | double num = argv[i].toNumber(); |
188 | if (e->hasException) |
189 | return Encode::undefined(); |
190 | int cp = static_cast<int>(num); |
191 | if (cp != num || cp < 0 || cp > 0x10ffff) |
192 | return e->throwRangeError(QStringLiteral("String.fromCodePoint: argument out of range.")); |
193 | if (cp > 0xffff) { |
194 | *ch = QChar::highSurrogate(ucs4: cp); |
195 | ++ch; |
196 | *ch = QChar::lowSurrogate(ucs4: cp); |
197 | } else { |
198 | *ch = QChar(cp); |
199 | } |
200 | ++ch; |
201 | } |
202 | result.truncate(pos: ch - result.constData()); |
203 | return e->newString(s: result)->asReturnedValue(); |
204 | } |
205 | |
206 | ReturnedValue StringCtor::method_raw(const FunctionObject *f, const Value *, const Value *argv, int argc) |
207 | { |
208 | Scope scope(f); |
209 | if (!argc) |
210 | return scope.engine->throwTypeError(); |
211 | |
212 | ScopedObject cooked(scope, argv[0].toObject(e: scope.engine)); |
213 | if (!cooked) |
214 | return scope.engine->throwTypeError(); |
215 | ScopedString rawString(scope, scope.engine->newIdentifier(QStringLiteral("raw"))); |
216 | ScopedValue rawValue(scope, cooked->get(name: rawString)); |
217 | ScopedObject raw(scope, rawValue->toObject(e: scope.engine)); |
218 | if (scope.hasException()) |
219 | return Encode::undefined(); |
220 | |
221 | ++argv; |
222 | --argc; |
223 | |
224 | QString result; |
225 | uint literalSegments = raw->getLength(); |
226 | if (!literalSegments) |
227 | return scope.engine->id_empty()->asReturnedValue(); |
228 | |
229 | uint nextIndex = 0; |
230 | ScopedValue val(scope); |
231 | while (1) { |
232 | val = raw->get(idx: nextIndex); |
233 | result += val->toQString(); |
234 | if (scope.hasException()) |
235 | return Encode::undefined(); |
236 | if (nextIndex + 1 == literalSegments) |
237 | return scope.engine->newString(s: result)->asReturnedValue(); |
238 | |
239 | if (nextIndex < static_cast<uint>(argc)) |
240 | result += argv[nextIndex].toQString(); |
241 | if (scope.hasException()) |
242 | return Encode::undefined(); |
243 | ++nextIndex; |
244 | } |
245 | } |
246 | |
247 | void StringPrototype::init(ExecutionEngine *engine, Object *ctor) |
248 | { |
249 | Scope scope(engine); |
250 | ScopedObject o(scope); |
251 | |
252 | // need to set this once again, as these were not fully defined when creating the string proto |
253 | Heap::InternalClass *ic = scope.engine->classes[ExecutionEngine::Class_StringObject]->changePrototype(proto: scope.engine->objectPrototype()->d()); |
254 | d()->internalClass.set(e: scope.engine, newVal: ic); |
255 | d()->string.set(e: scope.engine, newVal: scope.engine->id_empty()->d()); |
256 | setProperty(engine: scope.engine, index: Heap::StringObject::LengthPropertyIndex, v: Value::fromInt32(i: 0)); |
257 | |
258 | ctor->defineReadonlyProperty(name: engine->id_prototype(), value: (o = this)); |
259 | ctor->defineReadonlyConfigurableProperty(name: engine->id_length(), value: Value::fromInt32(i: 1)); |
260 | ctor->defineDefaultProperty(QStringLiteral("fromCharCode"), code: StringCtor::method_fromCharCode, argumentCount: 1); |
261 | ctor->defineDefaultProperty(QStringLiteral("fromCodePoint"), code: StringCtor::method_fromCodePoint, argumentCount: 1); |
262 | ctor->defineDefaultProperty(QStringLiteral("raw"), code: StringCtor::method_raw, argumentCount: 1); |
263 | |
264 | defineDefaultProperty(QStringLiteral("constructor"), value: (o = ctor)); |
265 | defineDefaultProperty(name: engine->id_toString(), code: method_toString); |
266 | defineDefaultProperty(name: engine->id_valueOf(), code: method_toString); // valueOf and toString are identical |
267 | defineDefaultProperty(QStringLiteral("charAt"), code: method_charAt, argumentCount: 1); |
268 | defineDefaultProperty(QStringLiteral("charCodeAt"), code: method_charCodeAt, argumentCount: 1); |
269 | defineDefaultProperty(QStringLiteral("codePointAt"), code: method_codePointAt, argumentCount: 1); |
270 | defineDefaultProperty(QStringLiteral("concat"), code: method_concat, argumentCount: 1); |
271 | defineDefaultProperty(QStringLiteral("endsWith"), code: method_endsWith, argumentCount: 1); |
272 | defineDefaultProperty(QStringLiteral("indexOf"), code: method_indexOf, argumentCount: 1); |
273 | defineDefaultProperty(QStringLiteral("includes"), code: method_includes, argumentCount: 1); |
274 | defineDefaultProperty(QStringLiteral("lastIndexOf"), code: method_lastIndexOf, argumentCount: 1); |
275 | defineDefaultProperty(QStringLiteral("localeCompare"), code: method_localeCompare, argumentCount: 1); |
276 | defineDefaultProperty(QStringLiteral("match"), code: method_match, argumentCount: 1); |
277 | defineDefaultProperty(QStringLiteral("normalize"), code: method_normalize, argumentCount: 0); |
278 | defineDefaultProperty(QStringLiteral("padEnd"), code: method_padEnd, argumentCount: 1); |
279 | defineDefaultProperty(QStringLiteral("padStart"), code: method_padStart, argumentCount: 1); |
280 | defineDefaultProperty(QStringLiteral("repeat"), code: method_repeat, argumentCount: 1); |
281 | defineDefaultProperty(QStringLiteral("replace"), code: method_replace, argumentCount: 2); |
282 | defineDefaultProperty(QStringLiteral("search"), code: method_search, argumentCount: 1); |
283 | defineDefaultProperty(QStringLiteral("slice"), code: method_slice, argumentCount: 2); |
284 | defineDefaultProperty(QStringLiteral("split"), code: method_split, argumentCount: 2); |
285 | defineDefaultProperty(QStringLiteral("startsWith"), code: method_startsWith, argumentCount: 1); |
286 | defineDefaultProperty(QStringLiteral("substr"), code: method_substr, argumentCount: 2); |
287 | defineDefaultProperty(QStringLiteral("substring"), code: method_substring, argumentCount: 2); |
288 | defineDefaultProperty(QStringLiteral("toLowerCase"), code: method_toLowerCase); |
289 | defineDefaultProperty(QStringLiteral("toLocaleLowerCase"), code: method_toLocaleLowerCase); |
290 | defineDefaultProperty(QStringLiteral("toUpperCase"), code: method_toUpperCase); |
291 | defineDefaultProperty(QStringLiteral("toLocaleUpperCase"), code: method_toLocaleUpperCase); |
292 | defineDefaultProperty(QStringLiteral("trim"), code: method_trim); |
293 | defineDefaultProperty(name: engine->symbol_iterator(), code: method_iterator); |
294 | } |
295 | |
296 | static Heap::String *thisAsString(ExecutionEngine *v4, const QV4::Value *thisObject) |
297 | { |
298 | if (String *s = thisObject->stringValue()) |
299 | return s->d(); |
300 | if (const StringObject *thisString = thisObject->as<StringObject>()) |
301 | return thisString->d()->string; |
302 | return thisObject->toString(e: v4); |
303 | } |
304 | |
305 | static QString getThisString(ExecutionEngine *v4, const QV4::Value *thisObject) |
306 | { |
307 | if (String *s = thisObject->stringValue()) |
308 | return s->toQString(); |
309 | if (const StringObject *thisString = thisObject->as<StringObject>()) |
310 | return thisString->d()->string->toQString(); |
311 | if (thisObject->isUndefined() || thisObject->isNull()) { |
312 | v4->throwTypeError(); |
313 | return QString(); |
314 | } |
315 | return thisObject->toQString(); |
316 | } |
317 | |
318 | ReturnedValue StringPrototype::method_toString(const FunctionObject *b, const Value *thisObject, const Value *, int) |
319 | { |
320 | if (thisObject->isString()) |
321 | return thisObject->asReturnedValue(); |
322 | |
323 | ExecutionEngine *v4 = b->engine(); |
324 | const StringObject *o = thisObject->as<StringObject>(); |
325 | if (!o) |
326 | return v4->throwTypeError(); |
327 | return o->d()->string->asReturnedValue(); |
328 | } |
329 | |
330 | ReturnedValue StringPrototype::method_charAt(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
331 | { |
332 | ExecutionEngine *v4 = b->engine(); |
333 | const QString str = getThisString(v4, thisObject); |
334 | if (v4->hasException) |
335 | return QV4::Encode::undefined(); |
336 | |
337 | double pos = 0; |
338 | if (argc > 0) |
339 | pos = argv[0].toInteger(); |
340 | |
341 | QString result; |
342 | if (pos >= 0 && pos < str.size()) |
343 | result += str.at(i: pos); |
344 | |
345 | return Encode(v4->newString(s: result)); |
346 | } |
347 | |
348 | ReturnedValue StringPrototype::method_charCodeAt(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
349 | { |
350 | ExecutionEngine *v4 = b->engine(); |
351 | const QString str = getThisString(v4, thisObject); |
352 | if (v4->hasException) |
353 | return QV4::Encode::undefined(); |
354 | |
355 | double pos = 0; |
356 | if (argc > 0) |
357 | pos = argv[0].toInteger(); |
358 | |
359 | |
360 | if (pos >= 0 && pos < str.size()) |
361 | RETURN_RESULT(Encode(str.at(pos).unicode())); |
362 | |
363 | return Encode(qt_qnan()); |
364 | } |
365 | |
366 | ReturnedValue StringPrototype::method_codePointAt(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) |
367 | { |
368 | ExecutionEngine *v4 = f->engine(); |
369 | QString value = getThisString(v4, thisObject); |
370 | if (v4->hasException) |
371 | return QV4::Encode::undefined(); |
372 | |
373 | double index = argc ? argv[0].toInteger() : 0.0; |
374 | if (v4->hasException) |
375 | return QV4::Encode::undefined(); |
376 | |
377 | if (index < 0 || index >= value.size()) |
378 | return Encode::undefined(); |
379 | |
380 | uint first = value.at(i: index).unicode(); |
381 | if (QChar::isHighSurrogate(ucs4: first) && index + 1 < value.size()) { |
382 | uint second = value.at(i: index + 1).unicode(); |
383 | if (QChar::isLowSurrogate(ucs4: second)) |
384 | return Encode(QChar::surrogateToUcs4(high: first, low: second)); |
385 | } |
386 | return Encode(first); |
387 | } |
388 | |
389 | ReturnedValue StringPrototype::method_concat(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
390 | { |
391 | ExecutionEngine *v4 = b->engine(); |
392 | QString value = getThisString(v4, thisObject); |
393 | if (v4->hasException) |
394 | return QV4::Encode::undefined(); |
395 | |
396 | Scope scope(v4); |
397 | ScopedString s(scope); |
398 | for (int i = 0; i < argc; ++i) { |
399 | s = argv[i].toString(e: scope.engine); |
400 | if (v4->hasException) |
401 | return QV4::Encode::undefined(); |
402 | |
403 | Q_ASSERT(s->isString()); |
404 | value += s->toQString(); |
405 | } |
406 | |
407 | return Encode(v4->newString(s: value)); |
408 | } |
409 | |
410 | ReturnedValue StringPrototype::method_endsWith(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
411 | { |
412 | ExecutionEngine *v4 = b->engine(); |
413 | const QString value = getThisString(v4, thisObject); |
414 | if (v4->hasException) |
415 | return QV4::Encode::undefined(); |
416 | |
417 | if (argc && argv[0].as<RegExpObject>()) |
418 | return v4->throwTypeError(); |
419 | QString searchString = (argc ? argv[0] : Value::undefinedValue()).toQString(); |
420 | if (v4->hasException) |
421 | return Encode::undefined(); |
422 | |
423 | double pos = value.size(); |
424 | if (argc > 1) |
425 | pos = argv[1].toInteger(); |
426 | |
427 | if (pos == value.size()) |
428 | RETURN_RESULT(Encode(value.endsWith(searchString))); |
429 | |
430 | QStringView stringToSearch = QStringView{value}.left(n: pos); |
431 | return Encode(stringToSearch.endsWith(s: searchString)); |
432 | } |
433 | |
434 | ReturnedValue StringPrototype::method_indexOf(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
435 | { |
436 | ExecutionEngine *v4 = b->engine(); |
437 | const QString value = getThisString(v4, thisObject); |
438 | if (v4->hasException) |
439 | return QV4::Encode::undefined(); |
440 | |
441 | QString searchString = (argc ? argv[0] : Value::undefinedValue()).toQString(); |
442 | if (v4->hasException) |
443 | return Encode::undefined(); |
444 | |
445 | double pos = 0; |
446 | if (argc > 1) |
447 | pos = argv[1].toInteger(); |
448 | |
449 | int index = -1; |
450 | if (!value.isEmpty()) |
451 | index = value.indexOf(s: searchString, from: qMin(a: qMax(a: pos, b: 0.0), b: double(value.size()))); |
452 | |
453 | return Encode(index); |
454 | } |
455 | |
456 | ReturnedValue StringPrototype::method_includes(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
457 | { |
458 | ExecutionEngine *v4 = b->engine(); |
459 | const QString value = getThisString(v4, thisObject); |
460 | if (v4->hasException) |
461 | return QV4::Encode::undefined(); |
462 | |
463 | if (argc && argv[0].as<RegExpObject>()) |
464 | return v4->throwTypeError(); |
465 | QString searchString = (argc ? argv[0] : Value::undefinedValue()).toQString(); |
466 | if (v4->hasException) |
467 | return Encode::undefined(); |
468 | |
469 | double pos = 0; |
470 | if (argc > 1) { |
471 | const Value &posArg = argv[1]; |
472 | pos = posArg.toInteger(); |
473 | if (!posArg.isInteger() && posArg.isNumber() && qIsInf(d: posArg.toNumber())) |
474 | pos = value.size(); |
475 | } |
476 | |
477 | if (pos == 0) |
478 | RETURN_RESULT(Encode(value.contains(searchString))); |
479 | |
480 | QStringView stringToSearch = QStringView{value}.mid(pos); |
481 | return Encode(stringToSearch.contains(s: searchString)); |
482 | } |
483 | |
484 | ReturnedValue StringPrototype::method_lastIndexOf(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
485 | { |
486 | ExecutionEngine *v4 = b->engine(); |
487 | const QString value = getThisString(v4, thisObject); |
488 | if (v4->hasException) |
489 | return QV4::Encode::undefined(); |
490 | |
491 | QString searchString = (argc ? argv[0] : Value::undefinedValue()).toQString(); |
492 | if (v4->hasException) |
493 | return Encode::undefined(); |
494 | |
495 | double position = argc > 1 ? RuntimeHelpers::toNumber(value: argv[1]) : +qInf(); |
496 | if (std::isnan(x: position)) |
497 | position = +qInf(); |
498 | else |
499 | position = std::trunc(x: position); |
500 | |
501 | int pos = std::trunc(x: qMin(a: qMax(a: position, b: 0.0), b: double(value.size()))); |
502 | if (!searchString.isEmpty() && pos == value.size()) |
503 | --pos; |
504 | if (searchString.isNull() && pos == 0) |
505 | RETURN_RESULT(Encode(-1)); |
506 | int index = value.lastIndexOf(s: searchString, from: pos); |
507 | return Encode(index); |
508 | } |
509 | |
510 | ReturnedValue StringPrototype::method_localeCompare(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
511 | { |
512 | ExecutionEngine *v4 = b->engine(); |
513 | const QString value = getThisString(v4, thisObject); |
514 | if (v4->hasException) |
515 | return QV4::Encode::undefined(); |
516 | |
517 | const QString that = (argc ? argv[0] : Value::undefinedValue()).toQString(); |
518 | return Encode(QString::localeAwareCompare(s1: value, s2: that)); |
519 | } |
520 | |
521 | ReturnedValue StringPrototype::method_match(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
522 | { |
523 | ExecutionEngine *v4 = b->engine(); |
524 | if (thisObject->isNullOrUndefined()) |
525 | return v4->throwTypeError(); |
526 | |
527 | Scope scope(v4); |
528 | if (argc && !argv[0].isNullOrUndefined()) { |
529 | ScopedObject r(scope, argv[0].toObject(e: scope.engine)); |
530 | if (scope.hasException()) |
531 | return Encode::undefined(); |
532 | ScopedValue f(scope, r->get(name: scope.engine->symbol_match())); |
533 | if (!f->isNullOrUndefined()) { |
534 | ScopedFunctionObject fo(scope, f); |
535 | if (!fo) |
536 | return scope.engine->throwTypeError(); |
537 | return checkedResult(v4: scope.engine, result: fo->call(thisObject: r, argv: thisObject, argc: 1)); |
538 | } |
539 | } |
540 | |
541 | ScopedString s(scope, thisObject->toString(e: v4)); |
542 | if (v4->hasException) |
543 | return Encode::undefined(); |
544 | |
545 | Scoped<RegExpObject> that(scope, argc ? argv[0] : Value::undefinedValue()); |
546 | if (!that) { |
547 | // convert args[0] to a regexp |
548 | that = RegExpCtor::virtualCallAsConstructor(f: b, argv, argc, b); |
549 | if (v4->hasException) |
550 | return Encode::undefined(); |
551 | } |
552 | Q_ASSERT(!!that); |
553 | |
554 | ScopedFunctionObject match(scope, that->get(name: scope.engine->symbol_match())); |
555 | if (!match) |
556 | return scope.engine->throwTypeError(); |
557 | return checkedResult(v4: scope.engine, result: match->call(thisObject: that, argv: s, argc: 1)); |
558 | } |
559 | |
560 | ReturnedValue StringPrototype::method_normalize(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) |
561 | { |
562 | ExecutionEngine *v4 = f->engine(); |
563 | const QString value = getThisString(v4, thisObject); |
564 | if (v4->hasException) |
565 | return Encode::undefined(); |
566 | |
567 | QString::NormalizationForm form = QString::NormalizationForm_C; |
568 | if (argc >= 1 && !argv[0].isUndefined()) { |
569 | QString f = argv[0].toQString(); |
570 | if (v4->hasException) |
571 | return Encode::undefined(); |
572 | if (f == QLatin1String("NFC")) |
573 | form = QString::NormalizationForm_C; |
574 | else if (f == QLatin1String("NFD")) |
575 | form = QString::NormalizationForm_D; |
576 | else if (f == QLatin1String("NFKC")) |
577 | form = QString::NormalizationForm_KC; |
578 | else if (f == QLatin1String("NFKD")) |
579 | form = QString::NormalizationForm_KD; |
580 | else |
581 | return v4->throwRangeError(message: QLatin1String("String.prototype.normalize: Invalid normalization form.")); |
582 | } |
583 | QString normalized = value.normalized(mode: form); |
584 | return v4->newString(s: normalized)->asReturnedValue(); |
585 | } |
586 | |
587 | ReturnedValue StringPrototype::method_padEnd(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) |
588 | { |
589 | ExecutionEngine *v4 = f->engine(); |
590 | if (thisObject->isNullOrUndefined()) |
591 | return v4->throwTypeError(); |
592 | |
593 | Scope scope(v4); |
594 | ScopedString s(scope, thisAsString(v4, thisObject)); |
595 | if (v4->hasException) |
596 | return Encode::undefined(); |
597 | if (!argc) |
598 | return s->asReturnedValue(); |
599 | |
600 | double maxLen = argv[0].toInteger(); |
601 | if (maxLen <= s->d()->length()) |
602 | return s->asReturnedValue(); |
603 | QString fillString = (argc > 1 && !argv[1].isUndefined()) ? argv[1].toQString() : u" "_s; |
604 | if (v4->hasException) |
605 | return Encode::undefined(); |
606 | |
607 | if (fillString.isEmpty()) |
608 | return s->asReturnedValue(); |
609 | |
610 | QString padded = s->toQString(); |
611 | int oldLength = padded.size(); |
612 | int toFill = maxLen - oldLength; |
613 | padded.resize(size: maxLen); |
614 | QChar *ch = padded.data() + oldLength; |
615 | while (toFill) { |
616 | int copy = qMin(a: fillString.size(), b: toFill); |
617 | memcpy(dest: ch, src: fillString.constData(), n: copy*sizeof(QChar)); |
618 | toFill -= copy; |
619 | ch += copy; |
620 | } |
621 | *ch = QChar::Null; |
622 | |
623 | return v4->newString(s: padded)->asReturnedValue(); |
624 | } |
625 | |
626 | ReturnedValue StringPrototype::method_padStart(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) |
627 | { |
628 | ExecutionEngine *v4 = f->engine(); |
629 | if (thisObject->isNullOrUndefined()) |
630 | return v4->throwTypeError(); |
631 | |
632 | Scope scope(v4); |
633 | ScopedString s(scope, thisAsString(v4, thisObject)); |
634 | if (v4->hasException) |
635 | return Encode::undefined(); |
636 | if (!argc) |
637 | return s->asReturnedValue(); |
638 | |
639 | double maxLen = argv[0].toInteger(); |
640 | if (maxLen <= s->d()->length()) |
641 | return s->asReturnedValue(); |
642 | QString fillString = (argc > 1 && !argv[1].isUndefined()) ? argv[1].toQString() : u" "_s; |
643 | if (v4->hasException) |
644 | return Encode::undefined(); |
645 | |
646 | if (fillString.isEmpty()) |
647 | return s->asReturnedValue(); |
648 | |
649 | QString original = s->toQString(); |
650 | int oldLength = original.size(); |
651 | int toFill = maxLen - oldLength; |
652 | QString padded; |
653 | padded.resize(size: maxLen); |
654 | QChar *ch = padded.data(); |
655 | while (toFill) { |
656 | int copy = qMin(a: fillString.size(), b: toFill); |
657 | memcpy(dest: ch, src: fillString.constData(), n: copy*sizeof(QChar)); |
658 | toFill -= copy; |
659 | ch += copy; |
660 | } |
661 | memcpy(dest: ch, src: original.constData(), n: oldLength*sizeof(QChar)); |
662 | ch += oldLength; |
663 | *ch = QChar::Null; |
664 | |
665 | return v4->newString(s: padded)->asReturnedValue(); |
666 | } |
667 | |
668 | |
669 | ReturnedValue StringPrototype::method_repeat(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
670 | { |
671 | ExecutionEngine *v4 = b->engine(); |
672 | const QString value = getThisString(v4, thisObject); |
673 | if (v4->hasException) |
674 | return QV4::Encode::undefined(); |
675 | |
676 | double repeats = (argc ? argv[0] : Value::undefinedValue()).toInteger(); |
677 | |
678 | if (repeats < 0 || qIsInf(d: repeats)) |
679 | return v4->throwRangeError(message: QLatin1String("Invalid count value")); |
680 | |
681 | return Encode(v4->newString(s: value.repeated(times: int(repeats)))); |
682 | } |
683 | |
684 | static void appendReplacementString(QString *result, const QString &input, const QString& replaceValue, uint* matchOffsets, int captureCount) |
685 | { |
686 | result->reserve(asize: result->size() + replaceValue.size()); |
687 | for (int i = 0; i < replaceValue.size(); ++i) { |
688 | if (replaceValue.at(i) == QLatin1Char('$') && i < replaceValue.size() - 1) { |
689 | ushort ch = replaceValue.at(i: i + 1).unicode(); |
690 | uint substStart = JSC::Yarr::offsetNoMatch; |
691 | uint substEnd = JSC::Yarr::offsetNoMatch; |
692 | int skip = 0; |
693 | if (ch == '$') { |
694 | *result += QChar(ch); |
695 | ++i; |
696 | continue; |
697 | } else if (ch == '&') { |
698 | substStart = matchOffsets[0]; |
699 | substEnd = matchOffsets[1]; |
700 | skip = 1; |
701 | } else if (ch == '`') { |
702 | substStart = 0; |
703 | substEnd = matchOffsets[0]; |
704 | skip = 1; |
705 | } else if (ch == '\'') { |
706 | substStart = matchOffsets[1]; |
707 | substEnd = input.size(); |
708 | skip = 1; |
709 | } else if (ch >= '0' && ch <= '9') { |
710 | uint capture = ch - '0'; |
711 | skip = 1; |
712 | if (i < replaceValue.size() - 2) { |
713 | ch = replaceValue.at(i: i + 2).unicode(); |
714 | if (ch >= '0' && ch <= '9') { |
715 | uint c = capture*10 + ch - '0'; |
716 | if (c < static_cast<uint>(captureCount)) { |
717 | capture = c; |
718 | skip = 2; |
719 | } |
720 | } |
721 | } |
722 | if (capture > 0 && capture < static_cast<uint>(captureCount)) { |
723 | substStart = matchOffsets[capture * 2]; |
724 | substEnd = matchOffsets[capture * 2 + 1]; |
725 | } else { |
726 | skip = 0; |
727 | } |
728 | } |
729 | i += skip; |
730 | if (substStart != JSC::Yarr::offsetNoMatch && substEnd != JSC::Yarr::offsetNoMatch) |
731 | *result += QStringView{input}.mid(pos: substStart, n: substEnd - substStart); |
732 | else if (skip == 0) // invalid capture reference. Taken as literal value |
733 | *result += replaceValue.at(i); |
734 | } else { |
735 | *result += replaceValue.at(i); |
736 | } |
737 | } |
738 | } |
739 | |
740 | ReturnedValue StringPrototype::method_replace(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
741 | { |
742 | QString string; |
743 | if (const StringObject *thisString = thisObject->as<StringObject>()) |
744 | string = thisString->d()->string->toQString(); |
745 | else |
746 | string = thisObject->toQString(); |
747 | |
748 | int numCaptures = 0; |
749 | int numStringMatches = 0; |
750 | |
751 | uint allocatedMatchOffsets = 64; |
752 | uint _matchOffsets[64]; |
753 | uint *matchOffsets = _matchOffsets; |
754 | |
755 | Scope scope(b); |
756 | ScopedValue searchValue(scope, argc ? argv[0] : Value::undefinedValue()); |
757 | Scoped<RegExpObject> regExp(scope, searchValue); |
758 | if (regExp) { |
759 | uint offset = 0; |
760 | uint nMatchOffsets = 0; |
761 | |
762 | // We extract the pointer here to work around a compiler bug on Android. |
763 | Scoped<RegExp> re(scope, regExp->value()); |
764 | while (true) { |
765 | int oldSize = nMatchOffsets; |
766 | if (allocatedMatchOffsets < nMatchOffsets + re->captureCount() * 2) { |
767 | allocatedMatchOffsets = qMax(a: allocatedMatchOffsets * 2, b: nMatchOffsets + re->captureCount() * 2); |
768 | uint *newOffsets = (uint *)malloc(size: allocatedMatchOffsets*sizeof(uint)); |
769 | memcpy(dest: newOffsets, src: matchOffsets, n: nMatchOffsets*sizeof(uint)); |
770 | if (matchOffsets != _matchOffsets) |
771 | free(ptr: matchOffsets); |
772 | matchOffsets = newOffsets; |
773 | } |
774 | if (re->match(string, start: offset, matchOffsets: matchOffsets + oldSize) == JSC::Yarr::offsetNoMatch) { |
775 | nMatchOffsets = oldSize; |
776 | break; |
777 | } |
778 | nMatchOffsets += re->captureCount() * 2; |
779 | if (!regExp->global()) |
780 | break; |
781 | offset = qMax(a: offset + 1, b: matchOffsets[oldSize + 1]); |
782 | } |
783 | if (regExp->global()) { |
784 | regExp->setLastIndex(0); |
785 | if (scope.hasException()) |
786 | return Encode::undefined(); |
787 | } |
788 | numStringMatches = nMatchOffsets / (regExp->value()->captureCount() * 2); |
789 | numCaptures = regExp->value()->captureCount(); |
790 | } else { |
791 | numCaptures = 1; |
792 | QString searchString = searchValue->toQString(); |
793 | int idx = string.indexOf(s: searchString); |
794 | if (idx != -1) { |
795 | numStringMatches = 1; |
796 | matchOffsets[0] = idx; |
797 | matchOffsets[1] = idx + searchString.size(); |
798 | } |
799 | } |
800 | |
801 | QString result; |
802 | ScopedValue replacement(scope); |
803 | ScopedValue replaceValue(scope, argc > 1 ? argv[1] : Value::undefinedValue()); |
804 | ScopedFunctionObject searchCallback(scope, replaceValue); |
805 | if (!!searchCallback) { |
806 | result.reserve(asize: string.size() + 10*numStringMatches); |
807 | ScopedValue entry(scope); |
808 | Value *arguments = scope.alloc(nValues: numCaptures + 2); |
809 | int lastEnd = 0; |
810 | for (int i = 0; i < numStringMatches; ++i) { |
811 | for (int k = 0; k < numCaptures; ++k) { |
812 | int idx = (i * numCaptures + k) * 2; |
813 | uint start = matchOffsets[idx]; |
814 | uint end = matchOffsets[idx + 1]; |
815 | entry = Value::undefinedValue(); |
816 | if (start != JSC::Yarr::offsetNoMatch && end != JSC::Yarr::offsetNoMatch) |
817 | entry = scope.engine->newString(s: string.mid(position: start, n: end - start)); |
818 | arguments[k] = entry; |
819 | } |
820 | uint matchStart = matchOffsets[i * numCaptures * 2]; |
821 | Q_ASSERT(matchStart >= static_cast<uint>(lastEnd)); |
822 | uint matchEnd = matchOffsets[i * numCaptures * 2 + 1]; |
823 | arguments[numCaptures] = Value::fromUInt32(i: matchStart); |
824 | arguments[numCaptures + 1] = scope.engine->newString(s: string); |
825 | |
826 | Value that = Value::undefinedValue(); |
827 | replacement = searchCallback->call(thisObject: &that, argv: arguments, argc: numCaptures + 2); |
828 | CHECK_EXCEPTION(); |
829 | result += QStringView{string}.mid(pos: lastEnd, n: matchStart - lastEnd); |
830 | result += replacement->toQString(); |
831 | lastEnd = matchEnd; |
832 | } |
833 | result += QStringView{string}.mid(pos: lastEnd); |
834 | } else { |
835 | QString newString = replaceValue->toQString(); |
836 | result.reserve(asize: string.size() + numStringMatches*newString.size()); |
837 | |
838 | int lastEnd = 0; |
839 | for (int i = 0; i < numStringMatches; ++i) { |
840 | int baseIndex = i * numCaptures * 2; |
841 | uint matchStart = matchOffsets[baseIndex]; |
842 | uint matchEnd = matchOffsets[baseIndex + 1]; |
843 | if (matchStart == JSC::Yarr::offsetNoMatch) |
844 | continue; |
845 | |
846 | result += QStringView{string}.mid(pos: lastEnd, n: matchStart - lastEnd); |
847 | appendReplacementString(result: &result, input: string, replaceValue: newString, matchOffsets: matchOffsets + baseIndex, captureCount: numCaptures); |
848 | lastEnd = matchEnd; |
849 | } |
850 | result += QStringView{string}.mid(pos: lastEnd); |
851 | } |
852 | |
853 | if (matchOffsets != _matchOffsets) |
854 | free(ptr: matchOffsets); |
855 | |
856 | return Encode(scope.engine->newString(s: result)); |
857 | } |
858 | |
859 | ReturnedValue StringPrototype::method_search(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
860 | { |
861 | Scope scope(b); |
862 | QString string = getThisString(v4: scope.engine, thisObject); |
863 | if (scope.hasException()) |
864 | return QV4::Encode::undefined(); |
865 | |
866 | Scoped<RegExpObject> regExp(scope, argc ? argv[0] : Value::undefinedValue()); |
867 | if (!regExp) { |
868 | regExp = scope.engine->regExpCtor()->callAsConstructor(argv, argc: 1); |
869 | if (scope.hasException()) |
870 | return QV4::Encode::undefined(); |
871 | |
872 | Q_ASSERT(regExp); |
873 | } |
874 | Scoped<RegExp> re(scope, regExp->value()); |
875 | Q_ALLOCA_VAR(uint, matchOffsets, regExp->value()->captureCount() * 2 * sizeof(uint)); |
876 | uint result = re->match(string, /*offset*/start: 0, matchOffsets); |
877 | if (result == JSC::Yarr::offsetNoMatch) |
878 | return Encode(-1); |
879 | else |
880 | return Encode(result); |
881 | } |
882 | |
883 | ReturnedValue StringPrototype::method_slice(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
884 | { |
885 | ExecutionEngine *v4 = b->engine(); |
886 | Scope scope(v4); |
887 | ScopedString s(scope, thisAsString(v4, thisObject)); |
888 | if (v4->hasException) |
889 | return QV4::Encode::undefined(); |
890 | Q_ASSERT(s); |
891 | |
892 | const double length = s->d()->length(); |
893 | |
894 | double start = argc ? argv[0].toInteger() : 0; |
895 | double end = (argc < 2 || argv[1].isUndefined()) |
896 | ? length : argv[1].toInteger(); |
897 | |
898 | if (start < 0) |
899 | start = qMax(a: length + start, b: 0.); |
900 | else |
901 | start = qMin(a: start, b: length); |
902 | |
903 | if (end < 0) |
904 | end = qMax(a: length + end, b: 0.); |
905 | else |
906 | end = qMin(a: end, b: length); |
907 | |
908 | const int intStart = int(start); |
909 | const int intEnd = int(end); |
910 | |
911 | int count = qMax(a: 0, b: intEnd - intStart); |
912 | return Encode(v4->memoryManager->alloc<ComplexString>(args: s->d(), args: intStart, args&: count)); |
913 | } |
914 | |
915 | ReturnedValue StringPrototype::method_split(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
916 | { |
917 | ExecutionEngine *v4 = b->engine(); |
918 | QString text = getThisString(v4, thisObject); |
919 | if (v4->hasException) |
920 | return QV4::Encode::undefined(); |
921 | |
922 | Scope scope(v4); |
923 | ScopedValue separatorValue(scope, argc ? argv[0] : Value::undefinedValue()); |
924 | ScopedValue limitValue(scope, argc > 1 ? argv[1] : Value::undefinedValue()); |
925 | |
926 | ScopedArrayObject array(scope, scope.engine->newArrayObject()); |
927 | |
928 | if (separatorValue->isUndefined()) { |
929 | if (limitValue->isUndefined()) { |
930 | ScopedString s(scope, scope.engine->newString(s: text)); |
931 | array->push_back(v: s); |
932 | return array.asReturnedValue(); |
933 | } |
934 | RETURN_RESULT(scope.engine->newString(text.left(limitValue->toInteger()))); |
935 | } |
936 | |
937 | uint limit = limitValue->isUndefined() ? UINT_MAX : limitValue->toUInt32(); |
938 | |
939 | if (limit == 0) |
940 | return array.asReturnedValue(); |
941 | |
942 | Scoped<RegExpObject> re(scope, separatorValue); |
943 | if (re) { |
944 | if (re->value()->pattern->isEmpty()) { |
945 | re = (RegExpObject *)nullptr; |
946 | separatorValue = scope.engine->newString(); |
947 | } |
948 | } |
949 | |
950 | ScopedString s(scope); |
951 | if (re) { |
952 | uint offset = 0; |
953 | Q_ALLOCA_VAR(uint, matchOffsets, re->value()->captureCount() * 2 * sizeof(uint)); |
954 | while (true) { |
955 | Scoped<RegExp> regexp(scope, re->value()); |
956 | uint result = regexp->match(string: text, start: offset, matchOffsets); |
957 | if (result == JSC::Yarr::offsetNoMatch) |
958 | break; |
959 | |
960 | array->push_back(v: (s = scope.engine->newString(s: text.mid(position: offset, n: matchOffsets[0] - offset)))); |
961 | offset = qMax(a: offset + 1, b: matchOffsets[1]); |
962 | |
963 | if (array->getLength() >= limit) |
964 | break; |
965 | |
966 | for (int i = 1; i < re->value()->captureCount(); ++i) { |
967 | uint start = matchOffsets[i * 2]; |
968 | uint end = matchOffsets[i * 2 + 1]; |
969 | array->push_back(v: (s = scope.engine->newString(s: text.mid(position: start, n: end - start)))); |
970 | if (array->getLength() >= limit) |
971 | break; |
972 | } |
973 | } |
974 | if (array->getLength() < limit) |
975 | array->push_back(v: (s = scope.engine->newString(s: text.mid(position: offset)))); |
976 | } else { |
977 | QString separator = separatorValue->toQString(); |
978 | if (separator.isEmpty()) { |
979 | for (uint i = 0; i < qMin(a: limit, b: uint(text.size())); ++i) |
980 | array->push_back(v: (s = scope.engine->newString(s: text.mid(position: i, n: 1)))); |
981 | return array.asReturnedValue(); |
982 | } |
983 | |
984 | int start = 0; |
985 | int end; |
986 | while ((end = text.indexOf(s: separator, from: start)) != -1) { |
987 | array->push_back(v: (s = scope.engine->newString(s: text.mid(position: start, n: end - start)))); |
988 | start = end + separator.size(); |
989 | if (array->getLength() >= limit) |
990 | break; |
991 | } |
992 | if (array->getLength() < limit && start != -1) |
993 | array->push_back(v: (s = scope.engine->newString(s: text.mid(position: start)))); |
994 | } |
995 | return array.asReturnedValue(); |
996 | } |
997 | |
998 | ReturnedValue StringPrototype::method_startsWith(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
999 | { |
1000 | ExecutionEngine *v4 = b->engine(); |
1001 | const QString value = getThisString(v4, thisObject); |
1002 | if (v4->hasException) |
1003 | return QV4::Encode::undefined(); |
1004 | |
1005 | if (argc && argv[0].as<RegExpObject>()) |
1006 | return v4->throwTypeError(); |
1007 | QString searchString = (argc ? argv[0] : Value::undefinedValue()).toQString(); |
1008 | if (v4->hasException) |
1009 | return Encode::undefined(); |
1010 | |
1011 | double pos = 0; |
1012 | if (argc > 1) |
1013 | pos = argv[1].toInteger(); |
1014 | |
1015 | pos = std::clamp( |
1016 | val: pos, |
1017 | lo: 0.0, |
1018 | hi: double(value.size())); |
1019 | |
1020 | if (pos == 0) |
1021 | return Encode(value.startsWith(s: searchString)); |
1022 | |
1023 | QStringView stringToSearch = QStringView{value}.mid(pos); |
1024 | RETURN_RESULT(Encode(stringToSearch.startsWith(searchString))); |
1025 | } |
1026 | |
1027 | ReturnedValue StringPrototype::method_substr(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
1028 | { |
1029 | ExecutionEngine *v4 = b->engine(); |
1030 | const QString value = getThisString(v4, thisObject); |
1031 | if (v4->hasException) |
1032 | return QV4::Encode::undefined(); |
1033 | |
1034 | double start = 0; |
1035 | if (argc > 0) |
1036 | start = argv[0].toInteger(); |
1037 | |
1038 | double length = +qInf(); |
1039 | if (argc > 1) |
1040 | length = argv[1].toInteger(); |
1041 | |
1042 | double count = value.size(); |
1043 | if (start < 0) |
1044 | start = qMax(a: count + start, b: 0.0); |
1045 | |
1046 | length = qMin(a: qMax(a: length, b: 0.0), b: count - start); |
1047 | |
1048 | qint32 x = Value::toInt32(d: start); |
1049 | qint32 y = Value::toInt32(d: length); |
1050 | return Encode(v4->newString(s: value.mid(position: x, n: y))); |
1051 | } |
1052 | |
1053 | ReturnedValue StringPrototype::method_substring(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
1054 | { |
1055 | ExecutionEngine *v4 = b->engine(); |
1056 | const QString value = getThisString(v4, thisObject); |
1057 | if (v4->hasException) |
1058 | return QV4::Encode::undefined(); |
1059 | |
1060 | int length = value.size(); |
1061 | |
1062 | double start = 0; |
1063 | double end = length; |
1064 | |
1065 | if (argc > 0) |
1066 | start = argv[0].toInteger(); |
1067 | |
1068 | if (argc > 1 && !argv[1].isUndefined()) |
1069 | end = argv[1].toInteger(); |
1070 | |
1071 | if (std::isnan(x: start) || start < 0) |
1072 | start = 0; |
1073 | |
1074 | if (std::isnan(x: end) || end < 0) |
1075 | end = 0; |
1076 | |
1077 | if (start > length) |
1078 | start = length; |
1079 | |
1080 | if (end > length) |
1081 | end = length; |
1082 | |
1083 | if (start > end) { |
1084 | double was = start; |
1085 | start = end; |
1086 | end = was; |
1087 | } |
1088 | |
1089 | qint32 x = (int)start; |
1090 | qint32 y = (int)(end - start); |
1091 | return Encode(v4->newString(s: value.mid(position: x, n: y))); |
1092 | } |
1093 | |
1094 | ReturnedValue StringPrototype::method_toLowerCase(const FunctionObject *b, const Value *thisObject, const Value *, int) |
1095 | { |
1096 | ExecutionEngine *v4 = b->engine(); |
1097 | const QString value = getThisString(v4, thisObject); |
1098 | if (v4->hasException) |
1099 | return QV4::Encode::undefined(); |
1100 | |
1101 | return Encode(v4->newString(s: value.toLower())); |
1102 | } |
1103 | |
1104 | ReturnedValue StringPrototype::method_toLocaleLowerCase(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
1105 | { |
1106 | return method_toLowerCase(b, thisObject, argv, argc); |
1107 | } |
1108 | |
1109 | ReturnedValue StringPrototype::method_toUpperCase(const FunctionObject *b, const Value *thisObject, const Value *, int) |
1110 | { |
1111 | ExecutionEngine *v4 = b->engine(); |
1112 | const QString value = getThisString(v4, thisObject); |
1113 | if (v4->hasException) |
1114 | return QV4::Encode::undefined(); |
1115 | |
1116 | return Encode(v4->newString(s: value.toUpper())); |
1117 | } |
1118 | |
1119 | ReturnedValue StringPrototype::method_toLocaleUpperCase(const FunctionObject *b, const Value *thisObject, const Value *argv, int argc) |
1120 | { |
1121 | return method_toUpperCase(b, thisObject, argv, argc); |
1122 | } |
1123 | |
1124 | ReturnedValue StringPrototype::method_trim(const FunctionObject *b, const Value *thisObject, const Value *, int) |
1125 | { |
1126 | ExecutionEngine *v4 = b->engine(); |
1127 | QString s = getThisString(v4, thisObject); |
1128 | if (v4->hasException) |
1129 | return QV4::Encode::undefined(); |
1130 | |
1131 | const QChar *chars = s.constData(); |
1132 | int start, end; |
1133 | for (start = 0; start < s.size(); ++start) { |
1134 | if (!chars[start].isSpace() && chars[start].unicode() != 0xfeff) |
1135 | break; |
1136 | } |
1137 | for (end = s.size() - 1; end >= start; --end) { |
1138 | if (!chars[end].isSpace() && chars[end].unicode() != 0xfeff) |
1139 | break; |
1140 | } |
1141 | |
1142 | return Encode(v4->newString(s: QString(chars + start, end - start + 1))); |
1143 | } |
1144 | |
1145 | |
1146 | |
1147 | ReturnedValue StringPrototype::method_iterator(const FunctionObject *b, const Value *thisObject, const Value *, int) |
1148 | { |
1149 | Scope scope(b); |
1150 | ScopedString s(scope, thisObject->toString(e: scope.engine)); |
1151 | if (!s || thisObject->isNullOrUndefined()) |
1152 | return scope.engine->throwTypeError(); |
1153 | |
1154 | Scoped<StringIteratorObject> si(scope, scope.engine->memoryManager->allocate<StringIteratorObject>(args: s->d(), args&: scope.engine)); |
1155 | return si->asReturnedValue(); |
1156 | } |
1157 |
Definitions
- init
- init
- getIndex
- length
- virtualDeleteProperty
- StringObjectOwnPropertyKeyIterator
- ~StringObjectOwnPropertyKeyIterator
- next
- virtualOwnPropertyKeys
- virtualGetOwnProperty
- init
- virtualCallAsConstructor
- virtualCall
- method_fromCharCode
- method_fromCodePoint
- method_raw
- init
- thisAsString
- getThisString
- method_toString
- method_charAt
- method_charCodeAt
- method_codePointAt
- method_concat
- method_endsWith
- method_indexOf
- method_includes
- method_lastIndexOf
- method_localeCompare
- method_match
- method_normalize
- method_padEnd
- method_padStart
- method_repeat
- appendReplacementString
- method_replace
- method_search
- method_slice
- method_split
- method_startsWith
- method_substr
- method_substring
- method_toLowerCase
- method_toLocaleLowerCase
- method_toUpperCase
- method_toLocaleUpperCase
- method_trim
Start learning QML with our Intro Training
Find out more