⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 b_string.c

📁 一个类似windows
💻 C
📖 第 1 页 / 共 2 页
字号:
	      || args[1].u.vbuiltin->info != ctx->regexp_info)
	    goto argument_type_error;

	  js_builtin_RegExp_search (vm, (char*)n->u.vstring->data,
	                            n->u.vstring->len, &args[1], result_return);
	}
      /* ***************************************************************** */
      else if (method == ctx->s_slice)
	{
	  int start, end;

	  if (args->u.vinteger != 1 && args->u.vinteger != 2)
	    goto argument_error;

	  if (args[1].type != JS_INTEGER)
	    goto argument_type_error;

	  start = args[1].u.vinteger;
	  if (start < 0)
	    start += n->u.vstring->len;
	  if (start < 0)
	    start = 0;
	  if (start > n->u.vstring->len)
	    start = n->u.vstring->len;

	  if (args->u.vinteger == 2)
	    {
	      if (args[2].type != JS_INTEGER)
		goto argument_type_error;

	      end = args[2].u.vinteger;
	      if (end < 0)
		end += n->u.vstring->len;
	      if (end < 0)
		end = 0;
	      if (end > n->u.vstring->len)
		end = n->u.vstring->len;
	    }
	  else
	    end = n->u.vstring->len;

	  if (start > end)
	    end = start;

	  js_vm_make_string (vm, result_return, (char*)n->u.vstring->data +
	                     start, end - start);
	}
      /* ***************************************************************** */
      else if (method == ctx->s_split)
	{
	  if (args->u.vinteger == 0)
	    {
	      js_vm_make_array (vm, result_return, 1);
	      js_vm_make_string (vm, &result_return->u.varray->data[0],
				 (char*)n->u.vstring->data, n->u.vstring->len);
	    }
	  else
	    {
	      unsigned int limit;

	      if (args->u.vinteger == 1)
		limit = -1;
	      else if (args->u.vinteger == 2)
		{
		  if (args[2].type != JS_INTEGER)
		    goto argument_type_error;

		  limit = args[2].u.vinteger;
		}
	      else
		goto argument_error;

	      if (args[1].type == JS_STRING)
		{
		  unsigned int start = 0, pos;
		  unsigned int alen = 0;

		  js_vm_make_array (vm, result_return, alen);

		  for (pos = 0;
		       (alen < limit
			&& pos + args[1].u.vstring->len <= n->u.vstring->len);
		       )
		    {
		      if (memcmp (n->u.vstring->data + pos,
				  args[1].u.vstring->data,
				  args[1].u.vstring->len) == 0)
			{
			  /* Found the separator. */
			  js_vm_expand_array (vm, result_return, alen + 1);
			  js_vm_make_string (vm,
					     &(result_return
					       ->u.varray->data[alen]),
					     (char*)n->u.vstring->data + start,
					     pos - start);
			  alen++;

			  if (args[1].u.vstring->len == 0)
			    {
			      start = pos;
			      pos++;
			    }
			  else
			    {
			      pos += args[1].u.vstring->len;
			      start = pos;
			    }
			}
		      else
			pos++;
		    }

		  if (alen < limit)
		    {
		      /* And finally, insert all leftovers. */
		      js_vm_expand_array (vm, result_return, alen + 1);
		      js_vm_make_string (vm,
					 &result_return->u.varray->data[alen],
					 (char*)n->u.vstring->data + start,
					 n->u.vstring->len - start);
		    }
		}
	      else if (args[1].type == JS_BUILTIN
		       && args[1].u.vbuiltin->info == ctx->regexp_info)
		{
		  js_builtin_RegExp_split (vm, (char*)n->u.vstring->data,
					   n->u.vstring->len, &args[1],
					   limit, result_return);
		}
	      else
		goto argument_type_error;
	    }
	}
      /* ***************************************************************** */
      else if (method == ctx->s_substr)
	{
	  int start, length;

	  if (args->u.vinteger != 1 && args->u.vinteger != 2)
	    goto argument_error;

	  if (args[1].type != JS_INTEGER)
	    goto argument_type_error;

	  start = args[1].u.vinteger;

	  if (args->u.vinteger == 2)
	    {
	      if (args[2].type != JS_INTEGER)
		goto argument_type_error;

	      length = args[2].u.vinteger;
	      if (length < 0)
		length = 0;
	    }
	  else
	    length = n->u.vstring->len;

	  if (start < 0)
	    start += n->u.vstring->len;
	  if (start < 0)
	    start = 0;
	  if (start > n->u.vstring->len)
	    start = n->u.vstring->len;

	  if (start + length > n->u.vstring->len)
	    length = n->u.vstring->len - start;

	  js_vm_make_string (vm, result_return, (char*)n->u.vstring->data +
	                     start, length);
	}
      /* ***************************************************************** */
      else if (method == ctx->s_substring)
	{
	  int start, end;

	  if (args->u.vinteger != 1 && args->u.vinteger != 2)
	    goto argument_error;

	  if (args[1].type != JS_INTEGER)
	    goto argument_type_error;

	  start = args[1].u.vinteger;

	  if (args->u.vinteger == 2)
	    {
	      if (args[2].type != JS_INTEGER)
		goto argument_type_error;

	      end = args[2].u.vinteger;
	    }
	  else
	    end = n->u.vstring->len;

	  if (start < 0)
	    start = 0;

	  if (end > n->u.vstring->len)
	    end = n->u.vstring->len;

	  if (start > end)
	    {
	      sprintf (vm->error,
		       "String.%s(): start index is bigger than end",
		       js_vm_symname (vm, method));
	      js_vm_error (vm);
	    }

	  js_vm_make_string (vm, result_return, (char*)n->u.vstring->data +
	                     start, end - start);
	}
      /* ***************************************************************** */
      else if (method == ctx->s_toLowerCase)
	{
	  if (args->u.vinteger != 0)
	    goto argument_type_error;

	  js_vm_make_string (vm, result_return, (char*)n->u.vstring->data,
			     n->u.vstring->len);

	  for (i = 0; i < result_return->u.vstring->len; i++)
	    result_return->u.vstring->data[i]
	      = js_latin1_tolower[result_return->u.vstring->data[i]];
	}
      /* ***************************************************************** */
      else if (method == ctx->s_toUpperCase)
	{
	  if (args->u.vinteger != 0)
	    goto argument_type_error;

	  js_vm_make_string (vm, result_return, (char*)n->u.vstring->data,
			     n->u.vstring->len);

	  for (i = 0; i < result_return->u.vstring->len; i++)
	    result_return->u.vstring->data[i]
	      = js_latin1_toupper[result_return->u.vstring->data[i]];
	}
      /* ***************************************************************** */
      else if (method == ctx->s_unpack)
	{
	  unsigned int op;
	  unsigned char *buffer;
	  unsigned int buflen;
	  unsigned int bufpos = 0;
	  JSUInt32 ui;
	  unsigned int result_len = 0;
	  JSNode *rnode;

	  if (args->u.vinteger != 1)
	    goto argument_error;
	  if (args[1].type != JS_STRING)
	    goto argument_type_error;

	  buffer = n->u.vstring->data;
	  buflen = n->u.vstring->len;

	  js_vm_make_array (vm, result_return, 0);

	  for (op = 0; op < args[1].u.vstring->len; op++)
	    {
	      switch (args[1].u.vstring->data[op])
		{
		case 'C':
		  UNPACK_NEED (1);
		  UNPACK_EXPAND ();
		  rnode->type = JS_INTEGER;
		  rnode->u.vinteger = buffer[bufpos++];
		  break;

		case 'n':
		  UNPACK_NEED (2);
		  UNPACK_EXPAND ();

		  ui = buffer[bufpos++];
		  ui <<= 8;
		  ui |= buffer[bufpos++];

		  rnode->type = JS_INTEGER;
		  rnode->u.vinteger = ui;
		  break;

		case 'N':
		  UNPACK_NEED (4);
		  UNPACK_EXPAND ();

		  ui = buffer[bufpos++];
		  ui <<= 8;
		  ui |= buffer[bufpos++];
		  ui <<= 8;
		  ui |= buffer[bufpos++];
		  ui <<= 8;
		  ui |= buffer[bufpos++];

		  rnode->type = JS_INTEGER;
		  rnode->u.vinteger = ui;
		  break;

		case 'd':
		  UNPACK_NEED (8);
		  UNPACK_EXPAND ();

		  rnode->type = JS_FLOAT;
		  memcpy (&rnode->u.vfloat, buffer + bufpos, 8);
		  bufpos += 8;
		  break;

		default:
		  /* Silently ignore it. */
		  break;
		}
	    }
	}
      /* ***************************************************************** */
      else
	return JS_PROPERTY_UNKNOWN;
    }
  else
    return JS_PROPERTY_UNKNOWN;

  return JS_PROPERTY_FOUND;


  /*
   * Error handling.
   */

 argument_error:
  sprintf (vm->error, "String.%s(): illegal amount of arguments",
	   js_vm_symname (vm, method));
  js_vm_error (vm);

 argument_type_error:
  sprintf (vm->error, "String %s(): illegal argument",
	   js_vm_symname (vm, method));
  js_vm_error (vm);

  /* NOTREACHED */
  return 0;
}


/* Property proc. */
static int
property (JSVirtualMachine *vm, JSBuiltinInfo *builtin_info,
	  void *instance_context, JSSymbol property, int set, JSNode *node)
{
  StringCtx *ctx = builtin_info->obj_context;
  JSNode *n = instance_context;

  if (n && property == ctx->s_length)
    {
      if (set)
	goto immutable;

      node->type = JS_INTEGER;
      node->u.vinteger = n->u.vstring->len;
    }
  else
    {
      if (!set)
	node->type = JS_UNDEFINED;

      return JS_PROPERTY_UNKNOWN;
    }

  return JS_PROPERTY_FOUND;


  /*
   * Error handling.
   */

 immutable:
  sprintf (vm->error, "String.%s: immutable property",
	   js_vm_symname (vm, property));
  js_vm_error (vm);

  /* NOTREACHED. */
  return 0;
}


/* New proc. */
static void
new_proc (JSVirtualMachine *vm, JSBuiltinInfo *builtin_info, JSNode *args,
	  JSNode *result_return)
{
  JSNode source_n;
  JSNode *source;

  if (args->u.vinteger == 0)
    js_vm_make_string (vm, result_return, NULL, 0);
  else if (args->u.vinteger == 1)
    {
      if (args[1].type == JS_STRING)
	source = &args[1];
      else
	{
	  js_vm_to_string (vm, &args[1], &source_n);
	  source = &source_n;
	}

      js_vm_make_string (vm, result_return, (char*)source->u.vstring->data,
			 source->u.vstring->len);
    }
  else
    {
      sprintf (vm->error, "new String(): illegal amount of arguments");
      js_vm_error (vm);
    }

  /* Set the [[Prototype]] and [[Class]] properties. */
  /* XXX 15.8.2 */
}

/*
 * Global functions.
 */

void
js_builtin_String (JSVirtualMachine *vm)
{
  StringCtx *ctx;
  JSNode *n;
  JSBuiltinInfo *info;

  ctx = js_calloc (vm, 1, sizeof (*ctx));

  ctx->s_length		= js_vm_intern (vm, "length");

  ctx->s_append		= js_vm_intern (vm, "append");
  ctx->s_charAt		= js_vm_intern (vm, "charAt");
  ctx->s_charCodeAt	= js_vm_intern (vm, "charCodeAt");
  ctx->s_concat		= js_vm_intern (vm, "concat");
  ctx->s_crc32		= js_vm_intern (vm, "crc32");
  ctx->s_fromCharCode	= js_vm_intern (vm, "fromCharCode");
  ctx->s_indexOf	= js_vm_intern (vm, "indexOf");
  ctx->s_lastIndexOf	= js_vm_intern (vm, "lastIndexOf");
  ctx->s_match		= js_vm_intern (vm, "match");
  ctx->s_pack		= js_vm_intern (vm, "pack");
  ctx->s_replace	= js_vm_intern (vm, "replace");
  ctx->s_search		= js_vm_intern (vm, "search");
  ctx->s_slice		= js_vm_intern (vm, "slice");
  ctx->s_split		= js_vm_intern (vm, "split");
  ctx->s_substr		= js_vm_intern (vm, "substr");
  ctx->s_substring	= js_vm_intern (vm, "substring");
  ctx->s_toLowerCase	= js_vm_intern (vm, "toLowerCase");
  ctx->s_toUpperCase	= js_vm_intern (vm, "toUpperCase");
  ctx->s_unpack		= js_vm_intern (vm, "unpack");

  info = js_vm_builtin_info_create (vm);
  vm->prim[JS_STRING] = info;

  info->global_method_proc	= global_method;
  info->method_proc 		= method;
  info->property_proc		= property;
  info->new_proc		= new_proc;
  info->obj_context		= ctx;
  info->obj_context_delete	= js_free;

  /* Define it. */
  n = &vm->globals[js_vm_intern (vm, "String")];
  js_vm_builtin_create (vm, n, info, NULL);

  /* Fetch the JSBuiltinInfo of the RegExp object. */
  n = &vm->globals[js_vm_intern (vm, "RegExp")];
  ctx->regexp_info = n->u.vbuiltin->info;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -