patch 9.1.1119: Vim9: Not able to use an autoloaded class from another autoloaded script
Problem:  Vim9: Not able to use an autoloaded class from another
          autoloaded script (Elliot)
Solution: make it work (Yegappan Lakshmanan)
fixes: #15031
closes: #16652
Signed-off-by: Yegappan Lakshmanan <yegappan@yahoo.com>
Signed-off-by: Christian Brabandt <cb@256bit.org>
			
			
This commit is contained in:
		
				
					committed by
					
						 Christian Brabandt
						Christian Brabandt
					
				
			
			
				
	
			
			
			
						parent
						
							b34a688ba0
						
					
				
				
					commit
					778ada48fb
				
			| @ -3162,34 +3162,53 @@ object_required_error(typval_T *tv) | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Accessing the member of an object stored in a variable of type "any". | ||||
|  * Accessing the variable or method of an object or a class stored in a | ||||
|  * variable of type "any". | ||||
|  * Returns OK if the member variable is present. | ||||
|  * Returns FAIL if the variable is not found. | ||||
|  */ | ||||
|     static int | ||||
| any_var_get_obj_member(class_T *current_class, isn_T *iptr, typval_T *tv) | ||||
| var_any_get_oc_member(class_T *current_class, isn_T *iptr, typval_T *tv) | ||||
| { | ||||
|     object_T	*obj = tv->vval.v_object; | ||||
|     int		is_object = tv->v_type == VAR_OBJECT; | ||||
|     class_T	*tv_cl; | ||||
|     object_T	*obj = NULL; | ||||
|     typval_T	mtv; | ||||
|  | ||||
|     if (obj == NULL) | ||||
|     if (is_object) | ||||
|     { | ||||
| 	SOURCING_LNUM = iptr->isn_lnum; | ||||
| 	emsg(_(e_using_null_object)); | ||||
| 	return FAIL; | ||||
| 	obj = tv->vval.v_object; | ||||
| 	if (obj == NULL) | ||||
| 	{ | ||||
| 	    SOURCING_LNUM = iptr->isn_lnum; | ||||
| 	    emsg(_(e_using_null_object)); | ||||
| 	    return FAIL; | ||||
| 	} | ||||
| 	tv_cl = obj->obj_class; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
| 	tv_cl = tv->vval.v_class; | ||||
| 	if (tv_cl == NULL) | ||||
| 	{ | ||||
| 	    SOURCING_LNUM = iptr->isn_lnum; | ||||
| 	    emsg(_(e_using_null_class)); | ||||
| 	    return FAIL; | ||||
| 	} | ||||
|     } | ||||
|  | ||||
|     // get_member_tv() needs the object information in the typval argument. | ||||
|     // So set the object information. | ||||
|     // get_member_tv() needs the class/object information in the typval | ||||
|     // argument.  So set the object information. | ||||
|     copy_tv(tv, &mtv); | ||||
|  | ||||
|     // 'name' can either be a object variable or a object method | ||||
|     // 'name' can either be an instance or class variable or method | ||||
|     int		namelen = (int)STRLEN(iptr->isn_arg.string); | ||||
|     int		save_did_emsg = did_emsg; | ||||
|  | ||||
|     if (get_member_tv(obj->obj_class, TRUE, iptr->isn_arg.string, namelen, | ||||
|     if (get_member_tv(tv_cl, is_object, iptr->isn_arg.string, namelen, | ||||
| 						current_class, &mtv) == OK) | ||||
|     { | ||||
| 	// instance or class variable | ||||
| 	copy_tv(&mtv, tv); | ||||
| 	clear_tv(&mtv); | ||||
| 	return OK; | ||||
| @ -3198,31 +3217,36 @@ any_var_get_obj_member(class_T *current_class, isn_T *iptr, typval_T *tv) | ||||
|     if (did_emsg != save_did_emsg) | ||||
| 	return FAIL; | ||||
|  | ||||
|     // could be a member function | ||||
|     ufunc_T	*obj_method; | ||||
|     int		obj_method_idx; | ||||
|     // could be a class or instance method | ||||
|     ufunc_T	*oc_method; | ||||
|     int		oc_method_idx; | ||||
|  | ||||
|     obj_method = method_lookup(obj->obj_class, VAR_OBJECT, | ||||
| 				iptr->isn_arg.string, namelen, | ||||
| 				&obj_method_idx); | ||||
|     if (obj_method == NULL) | ||||
|     oc_method = method_lookup(tv_cl, tv->v_type, iptr->isn_arg.string, | ||||
| 						namelen, &oc_method_idx); | ||||
|     if (oc_method == NULL) | ||||
|     { | ||||
| 	char	*msg; | ||||
|  | ||||
| 	SOURCING_LNUM = iptr->isn_lnum; | ||||
| 	semsg(_(e_variable_not_found_on_object_str_str), iptr->isn_arg.string, | ||||
| 		obj->obj_class->class_name); | ||||
| 	if (is_object) | ||||
| 	    msg = e_variable_not_found_on_object_str_str; | ||||
| 	else | ||||
| 	    msg = e_class_variable_str_not_found_in_class_str; | ||||
| 	semsg(_(msg), iptr->isn_arg.string, tv_cl->class_name); | ||||
| 	return FAIL; | ||||
|     } | ||||
|  | ||||
|     // Protected methods are not accessible outside the class | ||||
|     if (*obj_method->uf_name == '_' | ||||
| 			&& !class_instance_of(current_class, obj->obj_class)) | ||||
|     if (*oc_method->uf_name == '_' | ||||
| 			&& !class_instance_of(current_class, tv_cl)) | ||||
|     { | ||||
| 	semsg(_(e_cannot_access_protected_method_str), obj_method->uf_name); | ||||
| 	semsg(_(e_cannot_access_protected_method_str), oc_method->uf_name); | ||||
| 	return FAIL; | ||||
|     } | ||||
|  | ||||
|     // Create a partial for the member function | ||||
|     if (obj_method_to_partial_tv(obj, obj_method, tv) == FAIL) | ||||
|     // Create a partial for the instance or class method | ||||
|     if (obj_method_to_partial_tv(is_object ? obj : NULL, oc_method, tv) | ||||
| 								== FAIL) | ||||
| 	return FAIL; | ||||
|  | ||||
|     return OK; | ||||
| @ -5671,15 +5695,16 @@ exec_instructions(ectx_T *ectx) | ||||
|  | ||||
| 		    tv = STACK_TV_BOT(-1); | ||||
|  | ||||
| 		    if (tv->v_type == VAR_OBJECT) | ||||
| 		    if (tv->v_type == VAR_OBJECT | ||||
| 			    || tv->v_type == VAR_CLASS) | ||||
| 		    { | ||||
| 			if (dict_stack_save(tv) == FAIL) | ||||
| 			    goto on_fatal_error; | ||||
|  | ||||
| 			ufunc_T *ufunc = (((dfunc_T *)def_functions.ga_data) | ||||
| 					+ ectx->ec_dfunc_idx)->df_ufunc; | ||||
| 			// Class object (not a Dict) | ||||
| 			if (any_var_get_obj_member(ufunc->uf_class, iptr, tv) == FAIL) | ||||
| 			// Class or an object (not a Dict) | ||||
| 			if (var_any_get_oc_member(ufunc->uf_class, iptr, tv) == FAIL) | ||||
| 			    goto on_error; | ||||
| 		    } | ||||
| 		    else | ||||
|  | ||||
		Reference in New Issue
	
	Block a user