1 # This file is part of NIT (http://www.nitlanguage.org).
3 # Copyright 2014 Frédéric Vachon <fredvac@gmail.com>
4 # Copyright 2015 Alexis Laferrière <alexis.laf@xymus.net>
6 # Licensed under the Apache License, Version 2.0 (the "License");
7 # you may not use this file except in compliance with the License.
8 # You may obtain a copy of the License at
10 # http://www.apache.org/licenses/LICENSE-2.0
12 # Unless required by applicable law or agreed to in writing, software
13 # distributed under the License is distributed on an "AS IS" BASIS,
14 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 # See the License for the specific language governing permissions and
16 # limitations under the License.
18 # Contains the java and nit type representation used to convert java to nit code
21 import more_collections
23 import jtype_converter
26 private var converter
: JavaTypeConverter
27 var identifier
= new Array[String]
28 var generic_params
: nullable Array[JavaType] = null
31 # Has some generic type to be resolved (T extends foo => T is resolved to foo)
32 var has_unresolved_types
= false
34 # Dimension of primitive array: `int[][]` is 2d
35 var array_dimension
= 0
37 fun is_primitive_array
: Bool do return array_dimension
> 0
39 fun has_generic_params
: Bool do return not generic_params
== null
40 fun full_id
: String do return identifier
.join
(".")
41 fun id
: String do return identifier
.last
.replace
("$", "")
43 init(converter
: JavaTypeConverter) do self.converter
= converter
45 fun return_cast
: String do return converter
.cast_as_return
(self.id
)
47 fun param_cast
: String
49 if self.has_generic_params
then
50 return converter
.cast_as_param
(self.generic_params
[0].id
)
53 return converter
.cast_as_param
(self.id
)
56 fun to_nit_type
: NitType
61 if not is_primitive_array
then
62 type_id
= converter
.to_nit_type
(self.id
)
65 if type_id
== null then
66 nit_type
= self.extern_name
67 nit_type
.is_complete
= false
69 nit_type
= new NitType(type_id
)
72 if not self.has_generic_params
then return nit_type
74 nit_type
.generic_params
= new Array[NitType]
76 for param
in generic_params
do
77 var nit_param
= param
.to_nit_type
79 nit_type
.generic_params
.add
(nit_param
)
81 if not nit_param
.is_complete
then nit_type
.is_complete
= false
87 fun is_iterable
: Bool do return iterable
.has
(self.id
)
89 fun is_collection
: Bool do return is_primitive_array
or collections_list
.has
(self.id
)
91 fun is_wrapped
: Bool do return find_extern_class
!= null
93 fun extern_name
: NitType
95 if is_wrapped
then return new NitType.with_module
(find_extern_class
.as(not null).first
, find_extern_class
.as(not null).second
)
98 if is_primitive_array
then
99 # Primitive arrays have a special naming convention
100 name
= "Native" + extern_class_name
.join
.capitalized
+ "Array"
102 name
= "Native" + extern_class_name
.join
105 var nit_type
= new NitType(name
)
106 nit_type
.is_complete
= false
110 fun to_cast
(jtype
: String, is_param
: Bool): String
113 return converter
.cast_as_param
(jtype
)
116 return converter
.cast_as_return
(jtype
)
119 redef fun to_s
: String
121 var id
= self.full_id
123 if self.is_primitive_array
then
124 for i
in [0..array_dimension
[ do
127 else if self.has_generic_params
then
128 var gen_list
= new Array[String]
130 for param
in generic_params
do
131 gen_list
.add
(param
.to_s
)
134 id
+= "<{gen_list.join(", ")}>"
140 # To fully qualified package name
141 # Cuts the primitive array `[]`
142 fun to_package_name
: String
147 return str
.substring
(0, len
- (2*array_dimension
))
150 fun resolve_types
(conversion_map
: HashMap[String, Array[String]])
152 if identifier
.length
== 1 then
153 var resolved_id
= conversion_map
.get_or_null
(self.id
)
154 if resolved_id
!= null then self.identifier
= new Array[String].from
(resolved_id
)
157 if self.has_generic_params
then
158 for params
in generic_params
do params
.resolve_types
(conversion_map
)
162 private fun extern_class_name
: Array[String]
164 var class_name
= new Array[String]
165 class_name
.add
(self.id
)
167 if not self.has_generic_params
then return class_name
171 for param
in generic_params
do class_name
.add_all param
.extern_class_name
176 # Search inside `lib/android` directory for already wrapped classes
177 # If found, contains the class identifier and the Nit Module name
178 var find_extern_class
: nullable Couple[String, NitModule] is lazy
do
180 var regex
= "extern class [a-zA-Z1-9]\\\+[ ]\\\+in[ ]\\\+\"Java\
"[ ]*`\{[ ]*" + self.to_s + "\\\+[ ]*`\}"
181 var nit_dir
= "NIT_DIR".environ
182 var grep
= new ProcessReader("grep", "-r", regex
, nit_dir
/"lib/android/", nit_dir
/"lib/java/")
183 var to_eat
= ["private", "extern", "class"]
185 var output
= grep
.read_line
187 var output_class
= output
.substring_from
(output
.index_of
(':') + 1)
188 var tokens
= output_class
.split
(" ")
192 for token
in tokens
do
193 if to_eat
.has
(token
) then continue
198 if nclass_name
== "" then return null
200 var str
= output
.substring
(0, output
.search
(".nit").from
)
201 str
= str
.substring_from
(str
.last_index_of
('/') + 1)
202 var mod
= new NitModule(str
)
204 return new Couple[String, NitModule](nclass_name
, mod
)
207 # Comparison based on fully qualified named and generic params
208 # Ignores primitive array so `a.b.c[][] == a.b.c`
211 if other
isa JavaType then
212 return self.repr
== other
.repr
217 redef fun hash
do return self.repr
.hash
219 private fun repr
: String
221 var id
= self.full_id
223 if self.has_generic_params
then
224 var gen_list
= new Array[String]
226 for param
in generic_params
do
227 gen_list
.add
(param
.to_s
)
230 id
+= "<{gen_list.join(", ")}>"
236 var collections_list
: Array[String] is lazy
do return ["List", "ArrayList", "LinkedList", "Vector", "Set", "SortedSet", "HashSet", "TreeSet", "LinkedHashSet", "Map", "SortedMap", "HashMap", "TreeMap", "Hashtable", "LinkedHashMap"]
237 var iterable
: Array[String] is lazy
do return ["ArrayList", "Set", "HashSet", "LinkedHashSet", "LinkedList", "Stack", "TreeSet", "Vector"]
238 var maps
: Array[String] is lazy
do return ["Map", "SortedMap", "HashMap", "TreeMap", "Hashtable", "LinkedHashMap"]
242 var identifier
: String
244 var generic_params
: nullable Array[NitType] = null
246 # If this NitType was found in `lib/android`, contains the module name to import
247 var mod
: nullable NitModule
249 # Returns `true` if all types have been successfully converted to Nit type
250 var is_complete
: Bool = true
252 fun has_generic_params
: Bool do return not generic_params
== null
254 fun id
: String do return identifier
261 init with_generic_params
(id
: String, gen_params
: String...)
264 self.generic_params
= new Array[NitType]
265 for param
in gen_params
do self.generic_params
.add
new NitType(param
)
268 init with_module
(id
: String, mod
: NitModule)
274 redef fun to_s
: String
276 var id
= self.identifier
278 if self.has_generic_params
then
279 var gen_list
= new Array[String]
281 for param
in generic_params
do
282 gen_list
.add
(param
.to_s
)
285 id
+= "[{gen_list.join(", ")}]"
292 # Model of a single Java class
295 var class_type
= new JavaType(new JavaTypeConverter)
297 # Attributes of this class
298 var attributes
= new HashMap[String, JavaType]
300 # Methods of this class organized by their name
301 var methods
= new MultiHashMap[String, JavaMethod]
303 # Importations from this class
304 var imports
= new HashSet[NitModule]
307 # Model of all the Java class analyzed in one run
309 # Unknown Java types used in `classes`
310 var unknown_types
= new HashSet[JavaType]
312 # All analyzed classes
313 var classes
= new Array[JavaClass]
316 # A Java method, with its signature
318 # Type returned by the method
319 var return_type
: JavaType
321 # Type of the arguments of the method
322 var params
: Array[JavaType]
325 # A Nit module, use to import the referenced extern classes
330 redef fun ==(other
): Bool do return self.to_s
== other
.to_s
331 redef fun to_s
: String do return self.name
332 redef fun hash
: Int do return self.name
.hash