Importer-Exporter.js 50 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950
  1. (function(smalltalk,nil,_st){
  2. smalltalk.addPackage('Importer-Exporter');
  3. smalltalk.addClass('ChunkParser', smalltalk.Object, ['stream'], 'Importer-Exporter');
  4. smalltalk.ChunkParser.comment="I am responsible for parsing aStream contents in the chunk format.\x0a\x0a## API\x0a\x0a ChunkParser new\x0a stream: aStream;\x0a nextChunk";
  5. smalltalk.addMethod(
  6. smalltalk.method({
  7. selector: "nextChunk",
  8. category: 'reading',
  9. fn: function (){
  10. var self=this;
  11. var char,result,chunk;
  12. return smalltalk.withContext(function($ctx1) {
  13. var $1,$2,$3;
  14. var $early={};
  15. try {
  16. result=""._writeStream();
  17. _st((function(){
  18. return smalltalk.withContext(function($ctx2) {
  19. char=_st(self["@stream"])._next();
  20. char;
  21. return _st(char)._notNil();
  22. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}))._whileTrue_((function(){
  23. return smalltalk.withContext(function($ctx2) {
  24. $1=_st(char).__eq("!");
  25. if(smalltalk.assert($1)){
  26. $2=_st(_st(self["@stream"])._peek()).__eq("!");
  27. if(smalltalk.assert($2)){
  28. _st(self["@stream"])._next();
  29. } else {
  30. $3=_st(_st(result)._contents())._trimBoth();
  31. throw $early=[$3];
  32. };
  33. };
  34. return _st(result)._nextPut_(char);
  35. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}));
  36. return nil;
  37. }
  38. catch(e) {if(e===$early)return e[0]; throw e}
  39. }, function($ctx1) {$ctx1.fill(self,"nextChunk",{char:char,result:result,chunk:chunk},smalltalk.ChunkParser)})},
  40. args: [],
  41. source: "nextChunk\x0a\x09\x22The chunk format (Smalltalk Interchange Format or Fileout format)\x0a\x09is a trivial format but can be a bit tricky to understand:\x0a\x09\x09- Uses the exclamation mark as delimiter of chunks.\x0a\x09\x09- Inside a chunk a normal exclamation mark must be doubled.\x0a\x09\x09- A non empty chunk must be a valid Smalltalk expression.\x0a\x09\x09- A chunk on top level with a preceding empty chunk is an instruction chunk:\x0a\x09\x09\x09- The object created by the expression then takes over reading chunks.\x0a\x0a\x09This metod returns next chunk as a String (trimmed), empty String (all whitespace) or nil.\x22\x0a\x0a\x09| char result chunk |\x0a\x09result := '' writeStream.\x0a\x09\x09[char := stream next.\x0a\x09\x09char notNil] whileTrue: [\x0a\x09\x09\x09\x09char = '!' ifTrue: [\x0a\x09\x09\x09\x09\x09\x09stream peek = '!'\x0a\x09\x09\x09\x09\x09\x09\x09\x09ifTrue: [stream next \x22skipping the escape double\x22]\x0a\x09\x09\x09\x09\x09\x09\x09\x09ifFalse: [^result contents trimBoth \x22chunk end marker found\x22]].\x0a\x09\x09\x09\x09result nextPut: char].\x0a\x09^nil \x22a chunk needs to end with !\x22",
  42. messageSends: ["writeStream", "whileTrue:", "ifTrue:", "ifTrue:ifFalse:", "next", "trimBoth", "contents", "=", "peek", "nextPut:", "notNil"],
  43. referencedClasses: []
  44. }),
  45. smalltalk.ChunkParser);
  46. smalltalk.addMethod(
  47. smalltalk.method({
  48. selector: "stream:",
  49. category: 'accessing',
  50. fn: function (aStream){
  51. var self=this;
  52. return smalltalk.withContext(function($ctx1) {
  53. self["@stream"]=aStream;
  54. return self}, function($ctx1) {$ctx1.fill(self,"stream:",{aStream:aStream},smalltalk.ChunkParser)})},
  55. args: ["aStream"],
  56. source: "stream: aStream\x0a\x09stream := aStream",
  57. messageSends: [],
  58. referencedClasses: []
  59. }),
  60. smalltalk.ChunkParser);
  61. smalltalk.addMethod(
  62. smalltalk.method({
  63. selector: "on:",
  64. category: 'not yet classified',
  65. fn: function (aStream){
  66. var self=this;
  67. return smalltalk.withContext(function($ctx1) {
  68. var $1;
  69. $1=_st(self._new())._stream_(aStream);
  70. return $1;
  71. }, function($ctx1) {$ctx1.fill(self,"on:",{aStream:aStream},smalltalk.ChunkParser.klass)})},
  72. args: ["aStream"],
  73. source: "on: aStream\x0a\x09^self new stream: aStream",
  74. messageSends: ["stream:", "new"],
  75. referencedClasses: []
  76. }),
  77. smalltalk.ChunkParser.klass);
  78. smalltalk.addClass('Exporter', smalltalk.Object, [], 'Importer-Exporter');
  79. smalltalk.Exporter.comment="I am responsible for outputting Amber code into a JavaScript string.\x0a\x0aThe generated output is enough to reconstruct the exported data, including Smalltalk source code and other metadata.\x0a\x0a## Use case\x0a\x0aI am typically used to save code outside of the Amber runtime (committing to disk, etc.).\x0a\x0a## API\x0a\x0aUse `#exportAll`, `#exportClass:` or `#exportPackage:` methods.";
  80. smalltalk.addMethod(
  81. smalltalk.method({
  82. selector: "classNameFor:",
  83. category: 'private',
  84. fn: function (aClass){
  85. var self=this;
  86. return smalltalk.withContext(function($ctx1) {
  87. var $2,$3,$1;
  88. $2=_st(aClass)._isMetaclass();
  89. if(smalltalk.assert($2)){
  90. $1=_st(_st(_st(aClass)._instanceClass())._name()).__comma(".klass");
  91. } else {
  92. $3=_st(aClass)._isNil();
  93. if(smalltalk.assert($3)){
  94. $1="nil";
  95. } else {
  96. $1=_st(aClass)._name();
  97. };
  98. };
  99. return $1;
  100. }, function($ctx1) {$ctx1.fill(self,"classNameFor:",{aClass:aClass},smalltalk.Exporter)})},
  101. args: ["aClass"],
  102. source: "classNameFor: aClass\x0a\x09^aClass isMetaclass\x0a\x09\x09ifTrue: [aClass instanceClass name, '.klass']\x0a\x09\x09ifFalse: [\x0a\x09\x09aClass isNil\x0a\x09\x09\x09ifTrue: ['nil']\x0a\x09\x09\x09ifFalse: [aClass name]]",
  103. messageSends: ["ifTrue:ifFalse:", ",", "name", "instanceClass", "isNil", "isMetaclass"],
  104. referencedClasses: []
  105. }),
  106. smalltalk.Exporter);
  107. smalltalk.addMethod(
  108. smalltalk.method({
  109. selector: "exportAll",
  110. category: 'fileOut',
  111. fn: function (){
  112. var self=this;
  113. function $Smalltalk(){return smalltalk.Smalltalk||(typeof Smalltalk=="undefined"?nil:Smalltalk)}
  114. function $String(){return smalltalk.String||(typeof String=="undefined"?nil:String)}
  115. return smalltalk.withContext(function($ctx1) {
  116. var $1;
  117. $1=_st($String())._streamContents_((function(stream){
  118. return smalltalk.withContext(function($ctx2) {
  119. return _st(_st(_st($Smalltalk())._current())._packages())._do_((function(pkg){
  120. return smalltalk.withContext(function($ctx3) {
  121. return _st(stream)._nextPutAll_(self._exportPackage_(_st(pkg)._name()));
  122. }, function($ctx3) {$ctx3.fillBlock({pkg:pkg},$ctx2)})}));
  123. }, function($ctx2) {$ctx2.fillBlock({stream:stream},$ctx1)})}));
  124. return $1;
  125. }, function($ctx1) {$ctx1.fill(self,"exportAll",{},smalltalk.Exporter)})},
  126. args: [],
  127. source: "exportAll\x0a\x09\x22Export all packages in the system.\x22\x0a\x0a\x09^String streamContents: [:stream |\x0a\x09\x09Smalltalk current packages do: [:pkg |\x0a\x09\x09stream nextPutAll: (self exportPackage: pkg name)]]",
  128. messageSends: ["streamContents:", "do:", "nextPutAll:", "exportPackage:", "name", "packages", "current"],
  129. referencedClasses: ["Smalltalk", "String"]
  130. }),
  131. smalltalk.Exporter);
  132. smalltalk.addMethod(
  133. smalltalk.method({
  134. selector: "exportClass:",
  135. category: 'fileOut',
  136. fn: function (aClass){
  137. var self=this;
  138. function $String(){return smalltalk.String||(typeof String=="undefined"?nil:String)}
  139. return smalltalk.withContext(function($ctx1) {
  140. var $1;
  141. $1=_st($String())._streamContents_((function(stream){
  142. return smalltalk.withContext(function($ctx2) {
  143. self._exportDefinitionOf_on_(aClass,stream);
  144. self._exportMethodsOf_on_(aClass,stream);
  145. self._exportMetaDefinitionOf_on_(aClass,stream);
  146. return self._exportMethodsOf_on_(_st(aClass)._class(),stream);
  147. }, function($ctx2) {$ctx2.fillBlock({stream:stream},$ctx1)})}));
  148. return $1;
  149. }, function($ctx1) {$ctx1.fill(self,"exportClass:",{aClass:aClass},smalltalk.Exporter)})},
  150. args: ["aClass"],
  151. source: "exportClass: aClass\x0a\x09\x22Export a single class. Subclasses override these methods.\x22\x0a\x0a\x09^String streamContents: [:stream |\x0a\x09\x09self exportDefinitionOf: aClass on: stream.\x0a\x09\x09self exportMethodsOf: aClass on: stream.\x0a\x09\x09self exportMetaDefinitionOf: aClass on: stream.\x0a\x09\x09self exportMethodsOf: aClass class on: stream]",
  152. messageSends: ["streamContents:", "exportDefinitionOf:on:", "exportMethodsOf:on:", "exportMetaDefinitionOf:on:", "class"],
  153. referencedClasses: ["String"]
  154. }),
  155. smalltalk.Exporter);
  156. smalltalk.addMethod(
  157. smalltalk.method({
  158. selector: "exportDefinitionOf:on:",
  159. category: 'private',
  160. fn: function (aClass,aStream){
  161. var self=this;
  162. return smalltalk.withContext(function($ctx1) {
  163. var $1,$2,$3,$4,$5,$6,$7;
  164. $1=aStream;
  165. _st($1)._nextPutAll_("smalltalk.addClass(");
  166. _st($1)._nextPutAll_(_st("'".__comma(self._classNameFor_(aClass))).__comma("', "));
  167. _st($1)._nextPutAll_("smalltalk.".__comma(self._classNameFor_(_st(aClass)._superclass())));
  168. $2=_st($1)._nextPutAll_(", [");
  169. _st(_st(aClass)._instanceVariableNames())._do_separatedBy_((function(each){
  170. return smalltalk.withContext(function($ctx2) {
  171. return _st(aStream)._nextPutAll_(_st("'".__comma(each)).__comma("'"));
  172. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}),(function(){
  173. return smalltalk.withContext(function($ctx2) {
  174. return _st(aStream)._nextPutAll_(", ");
  175. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}));
  176. $3=aStream;
  177. _st($3)._nextPutAll_("], '");
  178. _st($3)._nextPutAll_(_st(_st(aClass)._category()).__comma("'"));
  179. $4=_st($3)._nextPutAll_(");");
  180. $5=_st(_st(aClass)._comment())._notEmpty();
  181. if(smalltalk.assert($5)){
  182. $6=aStream;
  183. _st($6)._lf();
  184. _st($6)._nextPutAll_("smalltalk.");
  185. _st($6)._nextPutAll_(self._classNameFor_(aClass));
  186. _st($6)._nextPutAll_(".comment=");
  187. _st($6)._nextPutAll_(_st(_st(aClass)._comment())._asJavascript());
  188. $7=_st($6)._nextPutAll_(";");
  189. $7;
  190. };
  191. _st(aStream)._lf();
  192. return self}, function($ctx1) {$ctx1.fill(self,"exportDefinitionOf:on:",{aClass:aClass,aStream:aStream},smalltalk.Exporter)})},
  193. args: ["aClass", "aStream"],
  194. source: "exportDefinitionOf: aClass on: aStream\x0a\x09aStream\x0a\x09\x09nextPutAll: 'smalltalk.addClass(';\x0a\x09\x09nextPutAll: '''', (self classNameFor: aClass), ''', ';\x0a\x09\x09nextPutAll: 'smalltalk.', (self classNameFor: aClass superclass);\x0a\x09\x09nextPutAll: ', ['.\x0a\x09aClass instanceVariableNames\x0a\x09\x09do: [:each | aStream nextPutAll: '''', each, '''']\x0a\x09\x09separatedBy: [aStream nextPutAll: ', '].\x0a\x09aStream\x0a\x09\x09nextPutAll: '], ''';\x0a\x09\x09nextPutAll: aClass category, '''';\x0a\x09\x09nextPutAll: ');'.\x0a\x09aClass comment notEmpty ifTrue: [\x0a\x09\x09aStream\x0a\x09\x09\x09lf;\x0a\x09\x09nextPutAll: 'smalltalk.';\x0a\x09\x09nextPutAll: (self classNameFor: aClass);\x0a\x09\x09nextPutAll: '.comment=';\x0a\x09\x09nextPutAll: aClass comment asJavascript;\x0a\x09\x09nextPutAll: ';'].\x0a\x09aStream lf",
  195. messageSends: ["nextPutAll:", ",", "classNameFor:", "superclass", "do:separatedBy:", "instanceVariableNames", "category", "ifTrue:", "lf", "asJavascript", "comment", "notEmpty"],
  196. referencedClasses: []
  197. }),
  198. smalltalk.Exporter);
  199. smalltalk.addMethod(
  200. smalltalk.method({
  201. selector: "exportMetaDefinitionOf:on:",
  202. category: 'private',
  203. fn: function (aClass,aStream){
  204. var self=this;
  205. function $String(){return smalltalk.String||(typeof String=="undefined"?nil:String)}
  206. return smalltalk.withContext(function($ctx1) {
  207. var $1,$2,$3;
  208. $1=_st(_st(_st(aClass)._class())._instanceVariableNames())._isEmpty();
  209. if(! smalltalk.assert($1)){
  210. $2=aStream;
  211. _st($2)._nextPutAll_("smalltalk.".__comma(self._classNameFor_(_st(aClass)._class())));
  212. $3=_st($2)._nextPutAll_(".iVarNames = [");
  213. $3;
  214. _st(_st(_st(aClass)._class())._instanceVariableNames())._do_separatedBy_((function(each){
  215. return smalltalk.withContext(function($ctx2) {
  216. return _st(aStream)._nextPutAll_(_st("'".__comma(each)).__comma("'"));
  217. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}),(function(){
  218. return smalltalk.withContext(function($ctx2) {
  219. return _st(aStream)._nextPutAll_(",");
  220. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}));
  221. _st(aStream)._nextPutAll_("];".__comma(_st($String())._lf()));
  222. };
  223. return self}, function($ctx1) {$ctx1.fill(self,"exportMetaDefinitionOf:on:",{aClass:aClass,aStream:aStream},smalltalk.Exporter)})},
  224. args: ["aClass", "aStream"],
  225. source: "exportMetaDefinitionOf: aClass on: aStream\x0a\x09aClass class instanceVariableNames isEmpty ifFalse: [\x0a\x09\x09aStream\x0a\x09\x09nextPutAll: 'smalltalk.', (self classNameFor: aClass class);\x0a\x09\x09nextPutAll: '.iVarNames = ['.\x0a\x09\x09aClass class instanceVariableNames\x0a\x09\x09do: [:each | aStream nextPutAll: '''', each, '''']\x0a\x09\x09separatedBy: [aStream nextPutAll: ','].\x0a\x09\x09aStream nextPutAll: '];', String lf]",
  226. messageSends: ["ifFalse:", "nextPutAll:", ",", "classNameFor:", "class", "do:separatedBy:", "instanceVariableNames", "lf", "isEmpty"],
  227. referencedClasses: ["String"]
  228. }),
  229. smalltalk.Exporter);
  230. smalltalk.addMethod(
  231. smalltalk.method({
  232. selector: "exportMethod:of:on:",
  233. category: 'private',
  234. fn: function (aMethod,aClass,aStream){
  235. var self=this;
  236. return smalltalk.withContext(function($ctx1) {
  237. var $1,$2,$3,$4;
  238. $1=aStream;
  239. _st($1)._nextPutAll_("smalltalk.addMethod(");
  240. _st($1)._lf();
  241. _st($1)._nextPutAll_("smalltalk.method({");
  242. _st($1)._lf();
  243. _st($1)._nextPutAll_(_st("selector: ".__comma(_st(_st(aMethod)._selector())._asJavascript())).__comma(","));
  244. _st($1)._lf();
  245. _st($1)._nextPutAll_(_st("category: '".__comma(_st(aMethod)._category())).__comma("',"));
  246. _st($1)._lf();
  247. _st($1)._nextPutAll_(_st("fn: ".__comma(_st(_st(aMethod)._fn())._compiledSource())).__comma(","));
  248. _st($1)._lf();
  249. _st($1)._nextPutAll_(_st("args: ".__comma(_st(_st(aMethod)._arguments())._asJavascript())).__comma(","));
  250. _st($1)._lf();
  251. _st($1)._nextPutAll_(_st("source: ".__comma(_st(_st(aMethod)._source())._asJavascript())).__comma(","));
  252. _st($1)._lf();
  253. _st($1)._nextPutAll_(_st("messageSends: ".__comma(_st(_st(aMethod)._messageSends())._asJavascript())).__comma(","));
  254. _st($1)._lf();
  255. $2=_st($1)._nextPutAll_("referencedClasses: ".__comma(_st(_st(aMethod)._referencedClasses())._asJavascript()));
  256. $3=aStream;
  257. _st($3)._lf();
  258. _st($3)._nextPutAll_("}),");
  259. _st($3)._lf();
  260. _st($3)._nextPutAll_("smalltalk.".__comma(self._classNameFor_(aClass)));
  261. _st($3)._nextPutAll_(");");
  262. _st($3)._lf();
  263. $4=_st($3)._lf();
  264. return self}, function($ctx1) {$ctx1.fill(self,"exportMethod:of:on:",{aMethod:aMethod,aClass:aClass,aStream:aStream},smalltalk.Exporter)})},
  265. args: ["aMethod", "aClass", "aStream"],
  266. source: "exportMethod: aMethod of: aClass on: aStream\x0a\x09aStream\x0a\x09\x09nextPutAll: 'smalltalk.addMethod(';lf;\x0a\x09\x09\x22nextPutAll: aMethod selector asSelector asJavascript, ',';lf;\x22\x0a\x09\x09nextPutAll: 'smalltalk.method({';lf;\x0a\x09\x09nextPutAll: 'selector: ', aMethod selector asJavascript, ',';lf;\x0a\x09\x09nextPutAll: 'category: ''', aMethod category, ''',';lf;\x0a\x09\x09nextPutAll: 'fn: ', aMethod fn compiledSource, ',';lf;\x0a\x09\x09nextPutAll: 'args: ', aMethod arguments asJavascript, ','; lf;\x0a\x09\x09nextPutAll: 'source: ', aMethod source asJavascript, ',';lf;\x0a\x09\x09nextPutAll: 'messageSends: ', aMethod messageSends asJavascript, ',';lf;\x0a\x09\x09nextPutAll: 'referencedClasses: ', aMethod referencedClasses asJavascript.\x0a\x09aStream\x0a\x09\x09lf;\x0a\x09\x09nextPutAll: '}),';lf;\x0a\x09\x09nextPutAll: 'smalltalk.', (self classNameFor: aClass);\x0a\x09\x09nextPutAll: ');';lf;lf",
  267. messageSends: ["nextPutAll:", "lf", ",", "asJavascript", "selector", "category", "compiledSource", "fn", "arguments", "source", "messageSends", "referencedClasses", "classNameFor:"],
  268. referencedClasses: []
  269. }),
  270. smalltalk.Exporter);
  271. smalltalk.addMethod(
  272. smalltalk.method({
  273. selector: "exportMethodsOf:on:",
  274. category: 'private',
  275. fn: function (aClass,aStream){
  276. var self=this;
  277. return smalltalk.withContext(function($ctx1) {
  278. var $1;
  279. _st(_st(_st(_st(aClass)._methodDictionary())._values())._sorted_((function(a,b){
  280. return smalltalk.withContext(function($ctx2) {
  281. return _st(_st(a)._selector()).__lt_eq(_st(b)._selector());
  282. }, function($ctx2) {$ctx2.fillBlock({a:a,b:b},$ctx1)})})))._do_((function(each){
  283. return smalltalk.withContext(function($ctx2) {
  284. $1=_st(_st(each)._category())._match_("^\x5c*");
  285. if(! smalltalk.assert($1)){
  286. return self._exportMethod_of_on_(each,aClass,aStream);
  287. };
  288. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}));
  289. _st(aStream)._lf();
  290. return self}, function($ctx1) {$ctx1.fill(self,"exportMethodsOf:on:",{aClass:aClass,aStream:aStream},smalltalk.Exporter)})},
  291. args: ["aClass", "aStream"],
  292. source: "exportMethodsOf: aClass on: aStream\x0a\x09\x22Issue #143: sort methods alphabetically\x22\x0a\x0a\x09((aClass methodDictionary values) sorted: [:a :b | a selector <= b selector]) do: [:each |\x0a\x09\x09(each category match: '^\x5c*') ifFalse: [\x0a\x09\x09\x09self exportMethod: each of: aClass on: aStream]].\x0a\x09aStream lf",
  293. messageSends: ["do:", "ifFalse:", "exportMethod:of:on:", "match:", "category", "sorted:", "<=", "selector", "values", "methodDictionary", "lf"],
  294. referencedClasses: []
  295. }),
  296. smalltalk.Exporter);
  297. smalltalk.addMethod(
  298. smalltalk.method({
  299. selector: "exportPackage:",
  300. category: 'fileOut',
  301. fn: function (packageName){
  302. var self=this;
  303. var package_;
  304. function $Smalltalk(){return smalltalk.Smalltalk||(typeof Smalltalk=="undefined"?nil:Smalltalk)}
  305. function $String(){return smalltalk.String||(typeof String=="undefined"?nil:String)}
  306. return smalltalk.withContext(function($ctx1) {
  307. var $1;
  308. $1=_st($String())._streamContents_((function(stream){
  309. return smalltalk.withContext(function($ctx2) {
  310. package_=_st(_st($Smalltalk())._current())._packageAt_(packageName);
  311. package_;
  312. self._exportPackageDefinitionOf_on_(package_,stream);
  313. _st(_st(_st(package_)._sortedClasses())._asSet())._do_((function(each){
  314. return smalltalk.withContext(function($ctx3) {
  315. return _st(stream)._nextPutAll_(self._exportClass_(each));
  316. }, function($ctx3) {$ctx3.fillBlock({each:each},$ctx2)})}));
  317. return self._exportPackageExtensionsOf_on_(package_,stream);
  318. }, function($ctx2) {$ctx2.fillBlock({stream:stream},$ctx1)})}));
  319. return $1;
  320. }, function($ctx1) {$ctx1.fill(self,"exportPackage:",{packageName:packageName,package_:package_},smalltalk.Exporter)})},
  321. args: ["packageName"],
  322. source: "exportPackage: packageName\x0a\x09\x22Export a given package by name.\x22\x0a\x0a\x09| package |\x0a\x09^String streamContents: [:stream |\x0a\x09\x09\x09\x09package := Smalltalk current packageAt: packageName.\x0a\x09\x09\x09\x09self exportPackageDefinitionOf: package on: stream.\x0a\x0a\x09\x09\x22Export classes in dependency order.\x0a\x09\x09Update (issue #171): Remove duplicates for export\x22\x0a\x09\x09\x09package sortedClasses asSet do: [:each |\x0a\x09\x09\x09\x09\x09\x09stream nextPutAll: (self exportClass: each)].\x0a\x09\x09self exportPackageExtensionsOf: package on: stream]",
  323. messageSends: ["streamContents:", "packageAt:", "current", "exportPackageDefinitionOf:on:", "do:", "nextPutAll:", "exportClass:", "asSet", "sortedClasses", "exportPackageExtensionsOf:on:"],
  324. referencedClasses: ["Smalltalk", "String"]
  325. }),
  326. smalltalk.Exporter);
  327. smalltalk.addMethod(
  328. smalltalk.method({
  329. selector: "exportPackageDefinitionOf:on:",
  330. category: 'private',
  331. fn: function (package_,aStream){
  332. var self=this;
  333. return smalltalk.withContext(function($ctx1) {
  334. var $1,$2;
  335. $1=aStream;
  336. _st($1)._nextPutAll_("smalltalk.addPackage(");
  337. _st($1)._nextPutAll_(_st("'".__comma(_st(package_)._name())).__comma("');"));
  338. $2=_st($1)._lf();
  339. return self}, function($ctx1) {$ctx1.fill(self,"exportPackageDefinitionOf:on:",{package_:package_,aStream:aStream},smalltalk.Exporter)})},
  340. args: ["package", "aStream"],
  341. source: "exportPackageDefinitionOf: package on: aStream\x0a\x09aStream\x0a\x09\x09nextPutAll: 'smalltalk.addPackage(';\x0a\x09\x09nextPutAll: '''', package name, ''');';\x0a\x09\x09lf",
  342. messageSends: ["nextPutAll:", ",", "name", "lf"],
  343. referencedClasses: []
  344. }),
  345. smalltalk.Exporter);
  346. smalltalk.addMethod(
  347. smalltalk.method({
  348. selector: "exportPackageExtensionsOf:on:",
  349. category: 'private',
  350. fn: function (package_,aStream){
  351. var self=this;
  352. var name;
  353. function $Smalltalk(){return smalltalk.Smalltalk||(typeof Smalltalk=="undefined"?nil:Smalltalk)}
  354. function $Package(){return smalltalk.Package||(typeof Package=="undefined"?nil:Package)}
  355. return smalltalk.withContext(function($ctx1) {
  356. var $1;
  357. name=_st(package_)._name();
  358. _st(_st($Package())._sortedClasses_(_st(_st($Smalltalk())._current())._classes()))._do_((function(each){
  359. return smalltalk.withContext(function($ctx2) {
  360. return _st([each,_st(each)._class()])._do_((function(aClass){
  361. return smalltalk.withContext(function($ctx3) {
  362. return _st(_st(_st(_st(aClass)._methodDictionary())._values())._sorted_((function(a,b){
  363. return smalltalk.withContext(function($ctx4) {
  364. return _st(_st(a)._selector()).__lt_eq(_st(b)._selector());
  365. }, function($ctx4) {$ctx4.fillBlock({a:a,b:b},$ctx3)})})))._do_((function(method){
  366. return smalltalk.withContext(function($ctx4) {
  367. $1=_st(_st(method)._category())._match_("^\x5c*".__comma(name));
  368. if(smalltalk.assert($1)){
  369. return self._exportMethod_of_on_(method,aClass,aStream);
  370. };
  371. }, function($ctx4) {$ctx4.fillBlock({method:method},$ctx3)})}));
  372. }, function($ctx3) {$ctx3.fillBlock({aClass:aClass},$ctx2)})}));
  373. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}));
  374. return self}, function($ctx1) {$ctx1.fill(self,"exportPackageExtensionsOf:on:",{package_:package_,aStream:aStream,name:name},smalltalk.Exporter)})},
  375. args: ["package", "aStream"],
  376. source: "exportPackageExtensionsOf: package on: aStream\x0a\x09\x22Issue #143: sort classes and methods alphabetically\x22\x0a\x0a\x09| name |\x0a\x09name := package name.\x0a\x09(Package sortedClasses: Smalltalk current classes) do: [:each |\x0a\x09\x09{each. each class} do: [:aClass |\x0a\x09\x09\x09((aClass methodDictionary values) sorted: [:a :b | a selector <= b selector]) do: [:method |\x0a\x09\x09\x09\x09(method category match: '^\x5c*', name) ifTrue: [\x0a\x09\x09\x09\x09\x09self exportMethod: method of: aClass on: aStream ]]]]",
  377. messageSends: ["name", "do:", "ifTrue:", "exportMethod:of:on:", "match:", ",", "category", "sorted:", "<=", "selector", "values", "methodDictionary", "class", "sortedClasses:", "classes", "current"],
  378. referencedClasses: ["Smalltalk", "Package"]
  379. }),
  380. smalltalk.Exporter);
  381. smalltalk.addClass('ChunkExporter', smalltalk.Exporter, [], 'Importer-Exporter');
  382. smalltalk.ChunkExporter.comment="I am an exporter dedicated to outputting Amber source code in the classic Smalltalk chunk format.\x0a\x0aI do not output any compiled code.";
  383. smalltalk.addMethod(
  384. smalltalk.method({
  385. selector: "chunkEscape:",
  386. category: 'private',
  387. fn: function (aString){
  388. var self=this;
  389. return smalltalk.withContext(function($ctx1) {
  390. var $1;
  391. $1=_st(_st(aString)._replace_with_("!","!!"))._trimBoth();
  392. return $1;
  393. }, function($ctx1) {$ctx1.fill(self,"chunkEscape:",{aString:aString},smalltalk.ChunkExporter)})},
  394. args: ["aString"],
  395. source: "chunkEscape: aString\x0a\x09\x22Replace all occurrences of ! with !! and trim at both ends.\x22\x0a\x0a\x09^(aString replace: '!' with: '!!') trimBoth",
  396. messageSends: ["trimBoth", "replace:with:"],
  397. referencedClasses: []
  398. }),
  399. smalltalk.ChunkExporter);
  400. smalltalk.addMethod(
  401. smalltalk.method({
  402. selector: "classNameFor:",
  403. category: 'private',
  404. fn: function (aClass){
  405. var self=this;
  406. return smalltalk.withContext(function($ctx1) {
  407. var $2,$3,$1;
  408. $2=_st(aClass)._isMetaclass();
  409. if(smalltalk.assert($2)){
  410. $1=_st(_st(_st(aClass)._instanceClass())._name()).__comma(" class");
  411. } else {
  412. $3=_st(aClass)._isNil();
  413. if(smalltalk.assert($3)){
  414. $1="nil";
  415. } else {
  416. $1=_st(aClass)._name();
  417. };
  418. };
  419. return $1;
  420. }, function($ctx1) {$ctx1.fill(self,"classNameFor:",{aClass:aClass},smalltalk.ChunkExporter)})},
  421. args: ["aClass"],
  422. source: "classNameFor: aClass\x0a\x09^aClass isMetaclass\x0a\x09\x09ifTrue: [aClass instanceClass name, ' class']\x0a\x09\x09ifFalse: [\x0a\x09\x09aClass isNil\x0a\x09\x09\x09ifTrue: ['nil']\x0a\x09\x09\x09ifFalse: [aClass name]]",
  423. messageSends: ["ifTrue:ifFalse:", ",", "name", "instanceClass", "isNil", "isMetaclass"],
  424. referencedClasses: []
  425. }),
  426. smalltalk.ChunkExporter);
  427. smalltalk.addMethod(
  428. smalltalk.method({
  429. selector: "exportDefinitionOf:on:",
  430. category: 'private',
  431. fn: function (aClass,aStream){
  432. var self=this;
  433. return smalltalk.withContext(function($ctx1) {
  434. var $1,$2,$3,$4,$5,$6,$7;
  435. $1=aStream;
  436. _st($1)._nextPutAll_(self._classNameFor_(_st(aClass)._superclass()));
  437. _st($1)._nextPutAll_(" subclass: #".__comma(self._classNameFor_(aClass)));
  438. _st($1)._lf();
  439. _st($1)._tab();
  440. $2=_st($1)._nextPutAll_("instanceVariableNames: '");
  441. _st(_st(aClass)._instanceVariableNames())._do_separatedBy_((function(each){
  442. return smalltalk.withContext(function($ctx2) {
  443. return _st(aStream)._nextPutAll_(each);
  444. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}),(function(){
  445. return smalltalk.withContext(function($ctx2) {
  446. return _st(aStream)._nextPutAll_(" ");
  447. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}));
  448. $3=aStream;
  449. _st($3)._nextPutAll_("'");
  450. _st($3)._lf();
  451. _st($3)._tab();
  452. _st($3)._nextPutAll_(_st("package: '".__comma(_st(aClass)._category())).__comma("'!"));
  453. $4=_st($3)._lf();
  454. $5=_st(_st(aClass)._comment())._notEmpty();
  455. if(smalltalk.assert($5)){
  456. $6=aStream;
  457. _st($6)._nextPutAll_(_st("!".__comma(self._classNameFor_(aClass))).__comma(" commentStamp!"));
  458. _st($6)._lf();
  459. _st($6)._nextPutAll_(_st(self._chunkEscape_(_st(aClass)._comment())).__comma("!"));
  460. $7=_st($6)._lf();
  461. $7;
  462. };
  463. _st(aStream)._lf();
  464. return self}, function($ctx1) {$ctx1.fill(self,"exportDefinitionOf:on:",{aClass:aClass,aStream:aStream},smalltalk.ChunkExporter)})},
  465. args: ["aClass", "aStream"],
  466. source: "exportDefinitionOf: aClass on: aStream\x0a\x09\x22Chunk format.\x22\x0a\x0a\x09aStream\x0a\x09\x09nextPutAll: (self classNameFor: aClass superclass);\x0a\x09\x09nextPutAll: ' subclass: #', (self classNameFor: aClass); lf;\x0a\x09\x09tab; nextPutAll: 'instanceVariableNames: '''.\x0a\x09aClass instanceVariableNames\x0a\x09\x09do: [:each | aStream nextPutAll: each]\x0a\x09\x09separatedBy: [aStream nextPutAll: ' '].\x0a\x09aStream\x0a\x09\x09nextPutAll: ''''; lf;\x0a\x09\x09tab; nextPutAll: 'package: ''', aClass category, '''!'; lf.\x0a\x09aClass comment notEmpty ifTrue: [\x0a\x09\x09aStream\x0a\x09\x09nextPutAll: '!', (self classNameFor: aClass), ' commentStamp!';lf;\x0a\x09\x09nextPutAll: (self chunkEscape: aClass comment), '!';lf].\x0a\x09aStream lf",
  467. messageSends: ["nextPutAll:", "classNameFor:", "superclass", ",", "lf", "tab", "do:separatedBy:", "instanceVariableNames", "category", "ifTrue:", "chunkEscape:", "comment", "notEmpty"],
  468. referencedClasses: []
  469. }),
  470. smalltalk.ChunkExporter);
  471. smalltalk.addMethod(
  472. smalltalk.method({
  473. selector: "exportMetaDefinitionOf:on:",
  474. category: 'private',
  475. fn: function (aClass,aStream){
  476. var self=this;
  477. return smalltalk.withContext(function($ctx1) {
  478. var $1,$2,$3,$4,$5;
  479. $1=_st(_st(_st(aClass)._class())._instanceVariableNames())._isEmpty();
  480. if(! smalltalk.assert($1)){
  481. $2=aStream;
  482. _st($2)._nextPutAll_(self._classNameFor_(_st(aClass)._class()));
  483. $3=_st($2)._nextPutAll_(" instanceVariableNames: '");
  484. $3;
  485. _st(_st(_st(aClass)._class())._instanceVariableNames())._do_separatedBy_((function(each){
  486. return smalltalk.withContext(function($ctx2) {
  487. return _st(aStream)._nextPutAll_(each);
  488. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}),(function(){
  489. return smalltalk.withContext(function($ctx2) {
  490. return _st(aStream)._nextPutAll_(" ");
  491. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}));
  492. $4=aStream;
  493. _st($4)._nextPutAll_("'!");
  494. _st($4)._lf();
  495. $5=_st($4)._lf();
  496. $5;
  497. };
  498. return self}, function($ctx1) {$ctx1.fill(self,"exportMetaDefinitionOf:on:",{aClass:aClass,aStream:aStream},smalltalk.ChunkExporter)})},
  499. args: ["aClass", "aStream"],
  500. source: "exportMetaDefinitionOf: aClass on: aStream\x0a\x0a\x09aClass class instanceVariableNames isEmpty ifFalse: [\x0a\x09\x09aStream\x0a\x09\x09\x09nextPutAll: (self classNameFor: aClass class);\x0a\x09\x09\x09nextPutAll: ' instanceVariableNames: '''.\x0a\x09\x09aClass class instanceVariableNames\x0a\x09\x09\x09do: [:each | aStream nextPutAll: each]\x0a\x09\x09\x09separatedBy: [aStream nextPutAll: ' '].\x0a\x09\x09aStream\x0a\x09\x09\x09nextPutAll: '''!'; lf; lf]",
  501. messageSends: ["ifFalse:", "nextPutAll:", "classNameFor:", "class", "do:separatedBy:", "instanceVariableNames", "lf", "isEmpty"],
  502. referencedClasses: []
  503. }),
  504. smalltalk.ChunkExporter);
  505. smalltalk.addMethod(
  506. smalltalk.method({
  507. selector: "exportMethod:of:on:",
  508. category: 'private',
  509. fn: function (aMethod,aClass,aStream){
  510. var self=this;
  511. return smalltalk.withContext(function($ctx1) {
  512. var $1,$2;
  513. $1=aStream;
  514. _st($1)._lf();
  515. _st($1)._lf();
  516. _st($1)._nextPutAll_(self._chunkEscape_(_st(aMethod)._source()));
  517. _st($1)._lf();
  518. $2=_st($1)._nextPutAll_("!");
  519. return self}, function($ctx1) {$ctx1.fill(self,"exportMethod:of:on:",{aMethod:aMethod,aClass:aClass,aStream:aStream},smalltalk.ChunkExporter)})},
  520. args: ["aMethod", "aClass", "aStream"],
  521. source: "exportMethod: aMethod of: aClass on: aStream\x0a\x09aStream\x0a\x09\x09lf; lf; nextPutAll: (self chunkEscape: aMethod source); lf;\x0a\x09\x09nextPutAll: '!'",
  522. messageSends: ["lf", "nextPutAll:", "chunkEscape:", "source"],
  523. referencedClasses: []
  524. }),
  525. smalltalk.ChunkExporter);
  526. smalltalk.addMethod(
  527. smalltalk.method({
  528. selector: "exportMethods:category:of:on:",
  529. category: 'private',
  530. fn: function (methods,category,aClass,aStream){
  531. var self=this;
  532. return smalltalk.withContext(function($ctx1) {
  533. var $1,$2,$3,$4;
  534. $1=aStream;
  535. _st($1)._nextPutAll_("!".__comma(self._classNameFor_(aClass)));
  536. $2=_st($1)._nextPutAll_(_st(" methodsFor: '".__comma(category)).__comma("'!"));
  537. _st(_st(methods)._sorted_((function(a,b){
  538. return smalltalk.withContext(function($ctx2) {
  539. return _st(_st(a)._selector()).__lt_eq(_st(b)._selector());
  540. }, function($ctx2) {$ctx2.fillBlock({a:a,b:b},$ctx1)})})))._do_((function(each){
  541. return smalltalk.withContext(function($ctx2) {
  542. return self._exportMethod_of_on_(each,aClass,aStream);
  543. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}));
  544. $3=aStream;
  545. _st($3)._nextPutAll_(" !");
  546. _st($3)._lf();
  547. $4=_st($3)._lf();
  548. return self}, function($ctx1) {$ctx1.fill(self,"exportMethods:category:of:on:",{methods:methods,category:category,aClass:aClass,aStream:aStream},smalltalk.ChunkExporter)})},
  549. args: ["methods", "category", "aClass", "aStream"],
  550. source: "exportMethods: methods category: category of: aClass on: aStream\x0a\x09\x22Issue #143: sort methods alphabetically\x22\x0a\x0a\x09aStream\x0a\x09\x09nextPutAll: '!', (self classNameFor: aClass);\x0a\x09\x09nextPutAll: ' methodsFor: ''', category, '''!'.\x0a\x09\x09(methods sorted: [:a :b | a selector <= b selector]) do: [:each |\x0a\x09\x09\x09\x09self exportMethod: each of: aClass on: aStream].\x0a\x09aStream nextPutAll: ' !'; lf; lf",
  551. messageSends: ["nextPutAll:", ",", "classNameFor:", "do:", "exportMethod:of:on:", "sorted:", "<=", "selector", "lf"],
  552. referencedClasses: []
  553. }),
  554. smalltalk.ChunkExporter);
  555. smalltalk.addMethod(
  556. smalltalk.method({
  557. selector: "exportMethodsOf:on:",
  558. category: 'private',
  559. fn: function (aClass,aStream){
  560. var self=this;
  561. var map;
  562. function $Dictionary(){return smalltalk.Dictionary||(typeof Dictionary=="undefined"?nil:Dictionary)}
  563. return smalltalk.withContext(function($ctx1) {
  564. var $1;
  565. map=_st($Dictionary())._new();
  566. _st(aClass)._protocolsDo_((function(category,methods){
  567. return smalltalk.withContext(function($ctx2) {
  568. $1=_st(category)._match_("^\x5c*");
  569. if(! smalltalk.assert($1)){
  570. return _st(map)._at_put_(category,methods);
  571. };
  572. }, function($ctx2) {$ctx2.fillBlock({category:category,methods:methods},$ctx1)})}));
  573. _st(_st(_st(map)._keys())._sorted_((function(a,b){
  574. return smalltalk.withContext(function($ctx2) {
  575. return _st(a).__lt_eq(b);
  576. }, function($ctx2) {$ctx2.fillBlock({a:a,b:b},$ctx1)})})))._do_((function(category){
  577. var methods;
  578. return smalltalk.withContext(function($ctx2) {
  579. methods=_st(map)._at_(category);
  580. methods;
  581. return self._exportMethods_category_of_on_(methods,category,aClass,aStream);
  582. }, function($ctx2) {$ctx2.fillBlock({category:category,methods:methods},$ctx1)})}));
  583. return self}, function($ctx1) {$ctx1.fill(self,"exportMethodsOf:on:",{aClass:aClass,aStream:aStream,map:map},smalltalk.ChunkExporter)})},
  584. args: ["aClass", "aStream"],
  585. source: "exportMethodsOf: aClass on: aStream\x0a\x09\x22Issue #143: sort protocol alphabetically\x22\x0a\x0a\x09| map |\x0a\x09map := Dictionary new.\x0a\x09aClass protocolsDo: [:category :methods |\x0a\x09\x09(category match: '^\x5c*') ifFalse: [ map at: category put: methods ]].\x0a\x09(map keys sorted: [:a :b | a <= b ]) do: [:category | | methods |\x0a\x09\x09methods := map at: category.\x0a\x09\x09self\x0a\x09\x09\x09exportMethods: methods\x0a\x09\x09\x09category: category\x0a\x09\x09\x09of: aClass\x0a\x09\x09\x09on: aStream ]",
  586. messageSends: ["new", "protocolsDo:", "ifFalse:", "at:put:", "match:", "do:", "at:", "exportMethods:category:of:on:", "sorted:", "<=", "keys"],
  587. referencedClasses: ["Dictionary"]
  588. }),
  589. smalltalk.ChunkExporter);
  590. smalltalk.addMethod(
  591. smalltalk.method({
  592. selector: "exportPackageDefinitionOf:on:",
  593. category: 'private',
  594. fn: function (package_,aStream){
  595. var self=this;
  596. return smalltalk.withContext(function($ctx1) {
  597. var $1,$2;
  598. $1=aStream;
  599. _st($1)._nextPutAll_(_st("Smalltalk current createPackage: '".__comma(_st(package_)._name())).__comma("'!"));
  600. $2=_st($1)._lf();
  601. return self}, function($ctx1) {$ctx1.fill(self,"exportPackageDefinitionOf:on:",{package_:package_,aStream:aStream},smalltalk.ChunkExporter)})},
  602. args: ["package", "aStream"],
  603. source: "exportPackageDefinitionOf: package on: aStream\x0a\x09\x22Chunk format.\x22\x0a\x0a\x09aStream\x0a\x09\x09nextPutAll: 'Smalltalk current createPackage: ''', package name, '''!';\x0a\x09\x09lf",
  604. messageSends: ["nextPutAll:", ",", "name", "lf"],
  605. referencedClasses: []
  606. }),
  607. smalltalk.ChunkExporter);
  608. smalltalk.addMethod(
  609. smalltalk.method({
  610. selector: "exportPackageExtensionsOf:on:",
  611. category: 'private',
  612. fn: function (package_,aStream){
  613. var self=this;
  614. var name,map;
  615. function $Dictionary(){return smalltalk.Dictionary||(typeof Dictionary=="undefined"?nil:Dictionary)}
  616. function $Smalltalk(){return smalltalk.Smalltalk||(typeof Smalltalk=="undefined"?nil:Smalltalk)}
  617. function $Package(){return smalltalk.Package||(typeof Package=="undefined"?nil:Package)}
  618. return smalltalk.withContext(function($ctx1) {
  619. var $1;
  620. name=_st(package_)._name();
  621. _st(_st($Package())._sortedClasses_(_st(_st($Smalltalk())._current())._classes()))._do_((function(each){
  622. return smalltalk.withContext(function($ctx2) {
  623. return _st([each,_st(each)._class()])._do_((function(aClass){
  624. return smalltalk.withContext(function($ctx3) {
  625. map=_st($Dictionary())._new();
  626. map;
  627. _st(aClass)._protocolsDo_((function(category,methods){
  628. return smalltalk.withContext(function($ctx4) {
  629. $1=_st(category)._match_("^\x5c*".__comma(name));
  630. if(smalltalk.assert($1)){
  631. return _st(map)._at_put_(category,methods);
  632. };
  633. }, function($ctx4) {$ctx4.fillBlock({category:category,methods:methods},$ctx3)})}));
  634. return _st(_st(_st(map)._keys())._sorted_((function(a,b){
  635. return smalltalk.withContext(function($ctx4) {
  636. return _st(a).__lt_eq(b);
  637. }, function($ctx4) {$ctx4.fillBlock({a:a,b:b},$ctx3)})})))._do_((function(category){
  638. var methods;
  639. return smalltalk.withContext(function($ctx4) {
  640. methods=_st(map)._at_(category);
  641. methods;
  642. return self._exportMethods_category_of_on_(methods,category,aClass,aStream);
  643. }, function($ctx4) {$ctx4.fillBlock({category:category,methods:methods},$ctx3)})}));
  644. }, function($ctx3) {$ctx3.fillBlock({aClass:aClass},$ctx2)})}));
  645. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}));
  646. return self}, function($ctx1) {$ctx1.fill(self,"exportPackageExtensionsOf:on:",{package_:package_,aStream:aStream,name:name,map:map},smalltalk.ChunkExporter)})},
  647. args: ["package", "aStream"],
  648. source: "exportPackageExtensionsOf: package on: aStream\x0a\x09\x22We need to override this one too since we need to group\x0a\x09all methods in a given protocol under a leading methodsFor: chunk\x0a\x09for that class.\x22\x0a\x0a\x09\x22Issue #143: sort protocol alphabetically\x22\x0a\x0a\x09| name map |\x0a\x09name := package name.\x0a\x09(Package sortedClasses: Smalltalk current classes) do: [:each |\x0a\x09\x09{each. each class} do: [:aClass |\x0a\x09\x09\x09map := Dictionary new.\x0a\x09\x09\x09aClass protocolsDo: [:category :methods |\x0a\x09\x09\x09\x09(category match: '^\x5c*', name) ifTrue: [ map at: category put: methods ]].\x0a\x09\x09\x09(map keys sorted: [:a :b | a <= b ]) do: [:category | | methods |\x0a\x09\x09\x09\x09methods := map at: category.\x0a\x09\x09\x09\x09self exportMethods: methods category: category of: aClass on: aStream ]]]",
  649. messageSends: ["name", "do:", "new", "protocolsDo:", "ifTrue:", "at:put:", "match:", ",", "at:", "exportMethods:category:of:on:", "sorted:", "<=", "keys", "class", "sortedClasses:", "classes", "current"],
  650. referencedClasses: ["Dictionary", "Smalltalk", "Package"]
  651. }),
  652. smalltalk.ChunkExporter);
  653. smalltalk.addClass('StrippedExporter', smalltalk.Exporter, [], 'Importer-Exporter');
  654. smalltalk.StrippedExporter.comment="I export Amber code into a JavaScript string, but without any optional associated data like the Amber source code.";
  655. smalltalk.addMethod(
  656. smalltalk.method({
  657. selector: "exportDefinitionOf:on:",
  658. category: 'private',
  659. fn: function (aClass,aStream){
  660. var self=this;
  661. return smalltalk.withContext(function($ctx1) {
  662. var $1,$2,$3,$4;
  663. $1=aStream;
  664. _st($1)._nextPutAll_("smalltalk.addClass(");
  665. _st($1)._nextPutAll_(_st("'".__comma(self._classNameFor_(aClass))).__comma("', "));
  666. _st($1)._nextPutAll_("smalltalk.".__comma(self._classNameFor_(_st(aClass)._superclass())));
  667. $2=_st($1)._nextPutAll_(", [");
  668. _st(_st(aClass)._instanceVariableNames())._do_separatedBy_((function(each){
  669. return smalltalk.withContext(function($ctx2) {
  670. return _st(aStream)._nextPutAll_(_st("'".__comma(each)).__comma("'"));
  671. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}),(function(){
  672. return smalltalk.withContext(function($ctx2) {
  673. return _st(aStream)._nextPutAll_(", ");
  674. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}));
  675. $3=aStream;
  676. _st($3)._nextPutAll_("], '");
  677. _st($3)._nextPutAll_(_st(_st(aClass)._category()).__comma("'"));
  678. $4=_st($3)._nextPutAll_(");");
  679. _st(aStream)._lf();
  680. return self}, function($ctx1) {$ctx1.fill(self,"exportDefinitionOf:on:",{aClass:aClass,aStream:aStream},smalltalk.StrippedExporter)})},
  681. args: ["aClass", "aStream"],
  682. source: "exportDefinitionOf: aClass on: aStream\x0a\x09aStream\x0a\x09\x09nextPutAll: 'smalltalk.addClass(';\x0a\x09\x09nextPutAll: '''', (self classNameFor: aClass), ''', ';\x0a\x09\x09nextPutAll: 'smalltalk.', (self classNameFor: aClass superclass);\x0a\x09\x09nextPutAll: ', ['.\x0a\x09aClass instanceVariableNames\x0a\x09\x09do: [:each | aStream nextPutAll: '''', each, '''']\x0a\x09\x09separatedBy: [aStream nextPutAll: ', '].\x0a\x09aStream\x0a\x09\x09nextPutAll: '], ''';\x0a\x09\x09nextPutAll: aClass category, '''';\x0a\x09\x09nextPutAll: ');'.\x0a\x09aStream lf",
  683. messageSends: ["nextPutAll:", ",", "classNameFor:", "superclass", "do:separatedBy:", "instanceVariableNames", "category", "lf"],
  684. referencedClasses: []
  685. }),
  686. smalltalk.StrippedExporter);
  687. smalltalk.addMethod(
  688. smalltalk.method({
  689. selector: "exportMethod:of:on:",
  690. category: 'private',
  691. fn: function (aMethod,aClass,aStream){
  692. var self=this;
  693. return smalltalk.withContext(function($ctx1) {
  694. var $1,$2;
  695. $1=aStream;
  696. _st($1)._nextPutAll_("smalltalk.addMethod(");
  697. _st($1)._lf();
  698. _st($1)._nextPutAll_("smalltalk.method({");
  699. _st($1)._lf();
  700. _st($1)._nextPutAll_(_st("selector: ".__comma(_st(_st(aMethod)._selector())._asJavascript())).__comma(","));
  701. _st($1)._lf();
  702. _st($1)._nextPutAll_(_st("fn: ".__comma(_st(_st(aMethod)._fn())._compiledSource())).__comma(","));
  703. _st($1)._lf();
  704. _st($1)._nextPutAll_("messageSends: ".__comma(_st(_st(aMethod)._messageSends())._asJavascript()));
  705. _st($1)._nextPutAll_("}),");
  706. _st($1)._lf();
  707. _st($1)._nextPutAll_("smalltalk.".__comma(self._classNameFor_(aClass)));
  708. _st($1)._nextPutAll_(");");
  709. _st($1)._lf();
  710. $2=_st($1)._lf();
  711. return self}, function($ctx1) {$ctx1.fill(self,"exportMethod:of:on:",{aMethod:aMethod,aClass:aClass,aStream:aStream},smalltalk.StrippedExporter)})},
  712. args: ["aMethod", "aClass", "aStream"],
  713. source: "exportMethod: aMethod of: aClass on: aStream\x0a\x09aStream\x0a\x09\x09nextPutAll: 'smalltalk.addMethod(';lf;\x0a\x09\x09\x22nextPutAll: aMethod selector asSelector asJavascript, ',';lf;\x22\x0a\x09\x09nextPutAll: 'smalltalk.method({';lf;\x0a\x09\x09nextPutAll: 'selector: ', aMethod selector asJavascript, ',';lf;\x0a\x09\x09nextPutAll: 'fn: ', aMethod fn compiledSource, ',';lf;\x0a\x09\x09nextPutAll: 'messageSends: ', aMethod messageSends asJavascript;\x0a\x09\x09nextPutAll: '}),';lf;\x0a\x09\x09nextPutAll: 'smalltalk.', (self classNameFor: aClass);\x0a\x09\x09nextPutAll: ');';lf;lf",
  714. messageSends: ["nextPutAll:", "lf", ",", "asJavascript", "selector", "compiledSource", "fn", "messageSends", "classNameFor:"],
  715. referencedClasses: []
  716. }),
  717. smalltalk.StrippedExporter);
  718. smalltalk.addClass('Importer', smalltalk.Object, [], 'Importer-Exporter');
  719. smalltalk.Importer.comment="I can import Amber code from a string in the chunk format.\x0a\x0a## API\x0a\x0a Importer new import: aString";
  720. smalltalk.addMethod(
  721. smalltalk.method({
  722. selector: "import:",
  723. category: 'fileIn',
  724. fn: function (aStream){
  725. var self=this;
  726. var chunk,result,parser,lastEmpty;
  727. function $ChunkParser(){return smalltalk.ChunkParser||(typeof ChunkParser=="undefined"?nil:ChunkParser)}
  728. function $Compiler(){return smalltalk.Compiler||(typeof Compiler=="undefined"?nil:Compiler)}
  729. return smalltalk.withContext(function($ctx1) {
  730. var $1,$2;
  731. parser=_st($ChunkParser())._on_(aStream);
  732. lastEmpty=false;
  733. _st((function(){
  734. return smalltalk.withContext(function($ctx2) {
  735. chunk=_st(parser)._nextChunk();
  736. chunk;
  737. return _st(chunk)._isNil();
  738. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}))._whileFalse_((function(){
  739. return smalltalk.withContext(function($ctx2) {
  740. $1=_st(chunk)._isEmpty();
  741. if(smalltalk.assert($1)){
  742. lastEmpty=true;
  743. return lastEmpty;
  744. } else {
  745. result=_st(_st($Compiler())._new())._evaluateExpression_(chunk);
  746. result;
  747. $2=lastEmpty;
  748. if(smalltalk.assert($2)){
  749. lastEmpty=false;
  750. lastEmpty;
  751. return _st(result)._scanFrom_(parser);
  752. };
  753. };
  754. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}));
  755. return self}, function($ctx1) {$ctx1.fill(self,"import:",{aStream:aStream,chunk:chunk,result:result,parser:parser,lastEmpty:lastEmpty},smalltalk.Importer)})},
  756. args: ["aStream"],
  757. source: "import: aStream\x0a\x09| chunk result parser lastEmpty |\x0a\x09parser := ChunkParser on: aStream.\x0a\x09lastEmpty := false.\x0a\x09[chunk := parser nextChunk.\x0a\x09chunk isNil] whileFalse: [\x0a\x09\x09chunk isEmpty\x0a\x09\x09\x09ifTrue: [lastEmpty := true]\x0a\x09\x09\x09ifFalse: [\x0a\x09\x09\x09\x09result := Compiler new evaluateExpression: chunk.\x0a\x09\x09\x09\x09lastEmpty\x0a\x09\x09\x09\x09\x09\x09ifTrue: [\x0a\x09\x09\x09\x09\x09\x09\x09\x09\x09lastEmpty := false.\x0a\x09\x09\x09\x09\x09\x09\x09\x09\x09result scanFrom: parser]]]",
  758. messageSends: ["on:", "whileFalse:", "ifTrue:ifFalse:", "evaluateExpression:", "new", "ifTrue:", "scanFrom:", "isEmpty", "nextChunk", "isNil"],
  759. referencedClasses: ["ChunkParser", "Compiler"]
  760. }),
  761. smalltalk.Importer);
  762. smalltalk.addClass('PackageHandler', smalltalk.Object, [], 'Importer-Exporter');
  763. smalltalk.PackageHandler.comment="I am responsible for handling package loading and committing.\x0a\x0aI should not be used directly. Instead, use the corresponding `Package` methods.";
  764. smalltalk.addMethod(
  765. smalltalk.method({
  766. selector: "ajaxPutAt:data:",
  767. category: 'private',
  768. fn: function (aURL,aString){
  769. var self=this;
  770. return smalltalk.withContext(function($ctx1) {
  771. _st(jQuery)._ajax_options_(aURL,smalltalk.HashedCollection._from_(["type".__minus_gt("PUT"),"data".__minus_gt(aString),"contentType".__minus_gt("text/plain;charset=UTF-8"),"error".__minus_gt((function(xhr){
  772. return smalltalk.withContext(function($ctx2) {
  773. return self._error_(_st(_st(_st("Commiting ".__comma(aURL)).__comma(" failed with reason: \x22")).__comma(_st(xhr)._responseText())).__comma("\x22"));
  774. }, function($ctx2) {$ctx2.fillBlock({xhr:xhr},$ctx1)})}))]));
  775. return self}, function($ctx1) {$ctx1.fill(self,"ajaxPutAt:data:",{aURL:aURL,aString:aString},smalltalk.PackageHandler)})},
  776. args: ["aURL", "aString"],
  777. source: "ajaxPutAt: aURL data: aString\x0a\x09jQuery\x0a\x09\x09ajax: aURL \x0a\x09\x09options: #{ \x0a\x09\x09\x09'type' -> 'PUT'.\x0a\x09\x09\x09'data' -> aString.\x0a\x09\x09\x09'contentType' -> 'text/plain;charset=UTF-8'.\x0a\x09\x09\x09'error' -> [ :xhr | self error: 'Commiting ' , aURL , ' failed with reason: \x22' , (xhr responseText) , '\x22'] }",
  778. messageSends: ["ajax:options:", "->", "error:", ",", "responseText"],
  779. referencedClasses: []
  780. }),
  781. smalltalk.PackageHandler);
  782. smalltalk.addMethod(
  783. smalltalk.method({
  784. selector: "commit:",
  785. category: 'committing',
  786. fn: function (aPackage){
  787. var self=this;
  788. function $Exporter(){return smalltalk.Exporter||(typeof Exporter=="undefined"?nil:Exporter)}
  789. function $StrippedExporter(){return smalltalk.StrippedExporter||(typeof StrippedExporter=="undefined"?nil:StrippedExporter)}
  790. function $ChunkExporter(){return smalltalk.ChunkExporter||(typeof ChunkExporter=="undefined"?nil:ChunkExporter)}
  791. return smalltalk.withContext(function($ctx1) {
  792. _st([_st($Exporter()).__minus_gt(_st(_st(_st(_st(aPackage)._commitPathJs()).__comma("/")).__comma(_st(aPackage)._name())).__comma(".js")),_st($StrippedExporter()).__minus_gt(_st(_st(_st(_st(aPackage)._commitPathJs()).__comma("/")).__comma(_st(aPackage)._name())).__comma(".deploy.js")),_st($ChunkExporter()).__minus_gt(_st(_st(_st(_st(aPackage)._commitPathSt()).__comma("/")).__comma(_st(aPackage)._name())).__comma(".st"))])._do_displayingProgress_((function(commitStrategy){
  793. var fileContents;
  794. return smalltalk.withContext(function($ctx2) {
  795. fileContents=_st(_st(_st(commitStrategy)._key())._new())._exportPackage_(_st(aPackage)._name());
  796. fileContents;
  797. return self._ajaxPutAt_data_(_st(commitStrategy)._value(),fileContents);
  798. }, function($ctx2) {$ctx2.fillBlock({commitStrategy:commitStrategy,fileContents:fileContents},$ctx1)})}),"Committing package ".__comma(_st(aPackage)._name()));
  799. return self}, function($ctx1) {$ctx1.fill(self,"commit:",{aPackage:aPackage},smalltalk.PackageHandler)})},
  800. args: ["aPackage"],
  801. source: "commit: aPackage\x0a\x09{ \x0a\x09\x09Exporter -> (aPackage commitPathJs, '/', aPackage name, '.js').\x0a\x09\x09StrippedExporter -> (aPackage commitPathJs, '/', aPackage name, '.deploy.js').\x0a\x09\x09ChunkExporter -> (aPackage commitPathSt, '/', aPackage name, '.st')\x0a\x09} \x0a\x09\x09do: [ :commitStrategy|| fileContents |\x0a\x09\x09\x09fileContents := (commitStrategy key new exportPackage: aPackage name).\x0a\x09\x09\x09self ajaxPutAt: commitStrategy value data: fileContents ]\x0a\x09\x09displayingProgress: 'Committing package ', aPackage name",
  802. messageSends: ["do:displayingProgress:", "exportPackage:", "name", "new", "key", "ajaxPutAt:data:", "value", ",", "->", "commitPathJs", "commitPathSt"],
  803. referencedClasses: ["Exporter", "StrippedExporter", "ChunkExporter"]
  804. }),
  805. smalltalk.PackageHandler);
  806. smalltalk.addMethod(
  807. smalltalk.method({
  808. selector: "loadPackage:prefix:",
  809. category: 'loading',
  810. fn: function (packageName,aString){
  811. var self=this;
  812. var url;
  813. return smalltalk.withContext(function($ctx1) {
  814. var $1;
  815. url=_st(_st(_st("/".__comma(aString)).__comma("/js/")).__comma(packageName)).__comma(".js");
  816. _st(jQuery)._ajax_options_(url,smalltalk.HashedCollection._from_(["type".__minus_gt("GET"),"dataType".__minus_gt("script"),"complete".__minus_gt((function(jqXHR,textStatus){
  817. return smalltalk.withContext(function($ctx2) {
  818. $1=_st(_st(jqXHR)._readyState()).__eq((4));
  819. if(smalltalk.assert($1)){
  820. return self._setupPackageNamed_prefix_(packageName,aString);
  821. };
  822. }, function($ctx2) {$ctx2.fillBlock({jqXHR:jqXHR,textStatus:textStatus},$ctx1)})})),"error".__minus_gt((function(){
  823. return smalltalk.withContext(function($ctx2) {
  824. return _st(window)._alert_("Could not load package at: ".__comma(url));
  825. }, function($ctx2) {$ctx2.fillBlock({},$ctx1)})}))]));
  826. return self}, function($ctx1) {$ctx1.fill(self,"loadPackage:prefix:",{packageName:packageName,aString:aString,url:url},smalltalk.PackageHandler)})},
  827. args: ["packageName", "aString"],
  828. source: "loadPackage: packageName prefix: aString\x0a\x09| url |\x0a\x09url := '/', aString, '/js/', packageName, '.js'.\x0a\x09jQuery\x0a\x09\x09ajax: url\x0a\x09\x09options: #{\x0a\x09\x09\x09'type' -> 'GET'.\x0a\x09\x09\x09'dataType' -> 'script'.\x0a\x09\x09\x09'complete' -> [ :jqXHR :textStatus |\x0a\x09\x09\x09\x09jqXHR readyState = 4\x0a\x09\x09\x09\x09\x09ifTrue: [ self setupPackageNamed: packageName prefix: aString ] ].\x0a\x09\x09\x09'error' -> [ window alert: 'Could not load package at: ', url ]\x0a\x09\x09}",
  829. messageSends: [",", "ajax:options:", "->", "ifTrue:", "setupPackageNamed:prefix:", "=", "readyState", "alert:"],
  830. referencedClasses: []
  831. }),
  832. smalltalk.PackageHandler);
  833. smalltalk.addMethod(
  834. smalltalk.method({
  835. selector: "loadPackages:prefix:",
  836. category: 'loading',
  837. fn: function (aCollection,aString){
  838. var self=this;
  839. return smalltalk.withContext(function($ctx1) {
  840. _st(aCollection)._do_((function(each){
  841. return smalltalk.withContext(function($ctx2) {
  842. return self._loadPackage_prefix_(each,aString);
  843. }, function($ctx2) {$ctx2.fillBlock({each:each},$ctx1)})}));
  844. return self}, function($ctx1) {$ctx1.fill(self,"loadPackages:prefix:",{aCollection:aCollection,aString:aString},smalltalk.PackageHandler)})},
  845. args: ["aCollection", "aString"],
  846. source: "loadPackages: aCollection prefix: aString\x0a\x09aCollection do: [ :each |\x0a\x09\x09self loadPackage: each prefix: aString ]",
  847. messageSends: ["do:", "loadPackage:prefix:"],
  848. referencedClasses: []
  849. }),
  850. smalltalk.PackageHandler);
  851. smalltalk.addMethod(
  852. smalltalk.method({
  853. selector: "setupPackageNamed:prefix:",
  854. category: 'private',
  855. fn: function (packageName,aString){
  856. var self=this;
  857. function $Package(){return smalltalk.Package||(typeof Package=="undefined"?nil:Package)}
  858. return smalltalk.withContext(function($ctx1) {
  859. var $1,$2;
  860. $1=_st($Package())._named_(packageName);
  861. _st($1)._setupClasses();
  862. _st($1)._commitPathJs_(_st("/".__comma(aString)).__comma("/js"));
  863. $2=_st($1)._commitPathSt_(_st("/".__comma(aString)).__comma("/st"));
  864. return self}, function($ctx1) {$ctx1.fill(self,"setupPackageNamed:prefix:",{packageName:packageName,aString:aString},smalltalk.PackageHandler)})},
  865. args: ["packageName", "aString"],
  866. source: "setupPackageNamed: packageName prefix: aString\x0a\x0a\x09(Package named: packageName)\x0a\x09\x09setupClasses;\x0a\x09\x09commitPathJs: '/', aString, '/js';\x0a\x09\x09commitPathSt: '/', aString, '/st'",
  867. messageSends: ["setupClasses", "named:", "commitPathJs:", ",", "commitPathSt:"],
  868. referencedClasses: ["Package"]
  869. }),
  870. smalltalk.PackageHandler);
  871. smalltalk.addMethod(
  872. smalltalk.method({
  873. selector: "loadPackages:prefix:",
  874. category: 'loading',
  875. fn: function (aCollection,aString){
  876. var self=this;
  877. return smalltalk.withContext(function($ctx1) {
  878. var $1;
  879. $1=_st(self._new())._loadPackages_prefix_(aCollection,aString);
  880. return $1;
  881. }, function($ctx1) {$ctx1.fill(self,"loadPackages:prefix:",{aCollection:aCollection,aString:aString},smalltalk.PackageHandler.klass)})},
  882. args: ["aCollection", "aString"],
  883. source: "loadPackages: aCollection prefix: aString\x0a\x09^ self new loadPackages: aCollection prefix: aString",
  884. messageSends: ["loadPackages:prefix:", "new"],
  885. referencedClasses: []
  886. }),
  887. smalltalk.PackageHandler.klass);
  888. smalltalk.addMethod(
  889. smalltalk.method({
  890. selector: "commit",
  891. category: '*Importer-Exporter',
  892. fn: function (){
  893. var self=this;
  894. function $PackageHandler(){return smalltalk.PackageHandler||(typeof PackageHandler=="undefined"?nil:PackageHandler)}
  895. return smalltalk.withContext(function($ctx1) {
  896. var $1;
  897. $1=_st(_st($PackageHandler())._new())._commit_(self);
  898. return $1;
  899. }, function($ctx1) {$ctx1.fill(self,"commit",{},smalltalk.Package)})},
  900. args: [],
  901. source: "commit\x0a\x09^ PackageHandler new commit: self",
  902. messageSends: ["commit:", "new"],
  903. referencedClasses: ["PackageHandler"]
  904. }),
  905. smalltalk.Package);
  906. })(smalltalk,nil,_st);