term_is_of_type(Term,Descr,PrintErrMsg) :-
is_inf(Term),!,
print_inf_error(Term,Descr,PrintErrMsg). %read(_Cont).
term_is_of_type(_Term,any,_PrintErrMsg) :-
!. /* anything is of type any */
term_is_of_type(Term,ground,_PrintErrMsg) :-
ground(Term),!.
term_is_of_type(Term,nonground,_PrintErrMsg) :-
\+(ground(Term)),!.
term_is_of_type(Term,var,_PrintErrMsg) :-
var(Term),!.
term_is_of_type(Term,nonvar,_PrintErrMsg) :-
nonvar(Term),!.
term_is_of_type(Term,integer,_PrintErrMsg) :-
integer(Term),!.
term_is_of_type(Term,float,_PrintErrMsg) :-
float(Term),!.
term_is_of_type(Term,number,_PrintErrMsg) :-
number(Term),!.
term_is_of_type(Term,atom,_PrintErrMsg) :-
atom(Term),!.
term_is_of_type(Term,atomic,_PrintErrMsg) :-
atomic(Term),!.
term_is_of_type(Term,compound,_PrintErrMsg) :-
compound(Term),!.
term_is_of_type(Term,simple,_PrintErrMsg) :-
\+ compound(Term),!.
term_is_of_type(Term,mutable,_PrintErrMsg) :-
mutable(Term),!.
term_is_of_type(Term,ask(Type),_PrintErrMsg) :-
!,
print('Type => '),print(Type),nl,
print('Term => '),print(Term),nl,
print('(y or n) =>'),read(UserChoice),
UserChoice=y.
term_is_of_type(Term,call(Module:Predicate),_PrintErrMsg) :-
functor(Call,Predicate,1),
arg(1,Call,Term),
call(Module:Call).
term_is_of_type(Term,list(Type),PrintErrMsg) :-
var(Term),!,
print_type_error(Term,list(Type),PrintErrMsg,term_is_of_type).
term_is_of_type([],list(_Type),_PrintErrMsg) :- !.
term_is_of_type([Head|Tail],list(Type),PrintErrMsg) :-
term_is_of_type(Head,Type,PrintErrMsg),!,
term_is_of_type(Tail,list(Type),yes).
term_is_of_type(Term,vlist(_Type),_PrintErrMsg) :-
var(Term),!. /* also to avoid modifying Term in the next 2 clauses */
term_is_of_type([],vlist(_Type),_PrintErrMsg) :- !.
term_is_of_type([Head|Tail],vlist(Type),PrintErrMsg) :-
term_is_of_type(Head,Type,PrintErrMsg),!,
term_is_of_type(Tail,vlist(Type),yes).
term_is_of_type(Term,term(Functor,ArgTypeList),PrintErrMsg) :-
var(Term),!,
print_type_error(Term,term(Functor,ArgTypeList),PrintErrMsg,term_is_of_type_var).
term_is_of_type(Term,term(Functor,ArgTypeList),_PrintErrMsg) :-
Term =.. [Functor|Args],!,
l_term_is_of_type(Args,ArgTypeList,yes), !.
term_is_of_type(Term,vterm(_Functor,_ArgTypeList),_PrintErrMsg) :-
var(Term),!. /* also to avoid modifying Term in the next clause */
term_is_of_type(Term,vterm(Functor,ArgTypeList),_PrintErrMsg) :-
Term =.. [Functor|Args],!,
l_term_is_of_type(Args,ArgTypeList,yes), !.
term_is_of_type(Term,Type,PrintErrMsg) :-
\+(predefined_type(Type)),
user_defined_type(Type,Descr), /* user defined type; we assume single definition exists */
!,
term_is_of_type(Term,Descr,PrintErrMsg).
term_is_of_type(Term,(Type1 ; Type2),PrintErrMsg) :-
(term_is_of_type(Term,Type1,no) -> true
; term_is_of_type(Term,Type2,PrintErrMsg)),
!.
term_is_of_type(Term,Type,PrintErrMsg) :-
print_type_error(Term,Type,PrintErrMsg,term_is_of_type_catchall).