CreditDisplay.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  1. /*global define*/
  2. define([
  3. '../Core/Credit',
  4. '../Core/defaultValue',
  5. '../Core/defined',
  6. '../Core/destroyObject',
  7. '../Core/DeveloperError'
  8. ], function(
  9. Credit,
  10. defaultValue,
  11. defined,
  12. destroyObject,
  13. DeveloperError) {
  14. "use strict";
  15. function displayTextCredit(credit, container, delimiter) {
  16. if (!defined(credit.element)) {
  17. var text = credit.text;
  18. var link = credit.link;
  19. var span = document.createElement('span');
  20. if (credit.hasLink()) {
  21. var a = document.createElement('a');
  22. a.textContent = text;
  23. a.href = link;
  24. a.target = '_blank';
  25. span.appendChild(a);
  26. } else {
  27. span.textContent = text;
  28. }
  29. span.className = 'cesium-credit-text';
  30. credit.element = span;
  31. }
  32. if (container.hasChildNodes()) {
  33. var del = document.createElement('span');
  34. del.textContent = delimiter;
  35. del.className = 'cesium-credit-delimiter';
  36. container.appendChild(del);
  37. }
  38. container.appendChild(credit.element);
  39. }
  40. function displayImageCredit(credit, container) {
  41. if (!defined(credit.element)) {
  42. var text = credit.text;
  43. var link = credit.link;
  44. var span = document.createElement('span');
  45. var content = document.createElement('img');
  46. content.src = credit.imageUrl;
  47. content.style['vertical-align'] = 'bottom';
  48. if (defined(text)) {
  49. content.alt = text;
  50. content.title = text;
  51. }
  52. if (credit.hasLink()) {
  53. var a = document.createElement('a');
  54. a.appendChild(content);
  55. a.href = link;
  56. a.target = '_blank';
  57. span.appendChild(a);
  58. } else {
  59. span.appendChild(content);
  60. }
  61. span.className = 'cesium-credit-image';
  62. credit.element = span;
  63. }
  64. container.appendChild(credit.element);
  65. }
  66. function contains(credits, credit) {
  67. var len = credits.length;
  68. for ( var i = 0; i < len; i++) {
  69. var existingCredit = credits[i];
  70. if (Credit.equals(existingCredit, credit)) {
  71. return true;
  72. }
  73. }
  74. return false;
  75. }
  76. function removeCreditDomElement(credit) {
  77. var element = credit.element;
  78. if (defined(element)) {
  79. var container = element.parentNode;
  80. if (!credit.hasImage()) {
  81. var delimiter = element.previousSibling;
  82. if (delimiter === null) {
  83. delimiter = element.nextSibling;
  84. }
  85. if (delimiter !== null) {
  86. container.removeChild(delimiter);
  87. }
  88. }
  89. container.removeChild(element);
  90. }
  91. }
  92. function displayTextCredits(creditDisplay, textCredits) {
  93. var i;
  94. var index;
  95. var credit;
  96. var displayedTextCredits = creditDisplay._displayedCredits.textCredits;
  97. for (i = 0; i < textCredits.length; i++) {
  98. credit = textCredits[i];
  99. if (defined(credit)) {
  100. index = displayedTextCredits.indexOf(credit);
  101. if (index === -1) {
  102. displayTextCredit(credit, creditDisplay._textContainer, creditDisplay._delimiter);
  103. } else {
  104. displayedTextCredits.splice(index, 1);
  105. }
  106. }
  107. }
  108. for (i = 0; i < displayedTextCredits.length; i++) {
  109. credit = displayedTextCredits[i];
  110. if (defined(credit)) {
  111. removeCreditDomElement(credit);
  112. }
  113. }
  114. }
  115. function displayImageCredits(creditDisplay, imageCredits) {
  116. var i;
  117. var index;
  118. var credit;
  119. var displayedImageCredits = creditDisplay._displayedCredits.imageCredits;
  120. for (i = 0; i < imageCredits.length; i++) {
  121. credit = imageCredits[i];
  122. if (defined(credit)) {
  123. index = displayedImageCredits.indexOf(credit);
  124. if (index === -1) {
  125. displayImageCredit(credit, creditDisplay._imageContainer);
  126. } else {
  127. displayedImageCredits.splice(index, 1);
  128. }
  129. }
  130. }
  131. for (i = 0; i < displayedImageCredits.length; i++) {
  132. credit = displayedImageCredits[i];
  133. if (defined(credit)) {
  134. removeCreditDomElement(credit);
  135. }
  136. }
  137. }
  138. /**
  139. * The credit display is responsible for displaying credits on screen.
  140. *
  141. * @param {HTMLElement} container The HTML element where credits will be displayed
  142. * @param {String} [delimiter= ' • '] The string to separate text credits
  143. *
  144. * @alias CreditDisplay
  145. * @constructor
  146. *
  147. * @example
  148. * var creditDisplay = new Cesium.CreditDisplay(creditContainer);
  149. */
  150. var CreditDisplay = function(container, delimiter) {
  151. //>>includeStart('debug', pragmas.debug);
  152. if (!defined(container)) {
  153. throw new DeveloperError('credit container is required');
  154. }
  155. //>>includeEnd('debug');
  156. var imageContainer = document.createElement('span');
  157. imageContainer.className = 'cesium-credit-imageContainer';
  158. var textContainer = document.createElement('span');
  159. textContainer.className = 'cesium-credit-textContainer';
  160. container.appendChild(imageContainer);
  161. container.appendChild(textContainer);
  162. this._delimiter = defaultValue(delimiter, ' • ');
  163. this._container = container;
  164. this._textContainer = textContainer;
  165. this._imageContainer = imageContainer;
  166. this._defaultImageCredits = [];
  167. this._defaultTextCredits = [];
  168. this._displayedCredits = {
  169. imageCredits : [],
  170. textCredits : []
  171. };
  172. this._currentFrameCredits = {
  173. imageCredits : [],
  174. textCredits : []
  175. };
  176. };
  177. /**
  178. * Adds a credit to the list of current credits to be displayed in the credit container
  179. *
  180. * @param {Credit} credit The credit to display
  181. */
  182. CreditDisplay.prototype.addCredit = function(credit) {
  183. //>>includeStart('debug', pragmas.debug);
  184. if (!defined(credit)) {
  185. throw new DeveloperError('credit must be defined');
  186. }
  187. //>>includeEnd('debug');
  188. if (credit.hasImage()) {
  189. var imageCredits = this._currentFrameCredits.imageCredits;
  190. if (!contains(this._defaultImageCredits, credit)) {
  191. imageCredits[credit.id] = credit;
  192. }
  193. } else {
  194. var textCredits = this._currentFrameCredits.textCredits;
  195. if (!contains(this._defaultTextCredits, credit)) {
  196. textCredits[credit.id] = credit;
  197. }
  198. }
  199. };
  200. /**
  201. * Adds credits that will persist until they are removed
  202. *
  203. * @param {Credit} credit The credit to added to defaults
  204. */
  205. CreditDisplay.prototype.addDefaultCredit = function(credit) {
  206. //>>includeStart('debug', pragmas.debug);
  207. if (!defined(credit)) {
  208. throw new DeveloperError('credit must be defined');
  209. }
  210. //>>includeEnd('debug');
  211. if (credit.hasImage()) {
  212. var imageCredits = this._defaultImageCredits;
  213. if (!contains(imageCredits, credit)) {
  214. imageCredits.push(credit);
  215. }
  216. } else {
  217. var textCredits = this._defaultTextCredits;
  218. if (!contains(textCredits, credit)) {
  219. textCredits.push(credit);
  220. }
  221. }
  222. };
  223. /**
  224. * Removes a default credit
  225. *
  226. * @param {Credit} credit The credit to be removed from defaults
  227. */
  228. CreditDisplay.prototype.removeDefaultCredit = function(credit) {
  229. //>>includeStart('debug', pragmas.debug);
  230. if (!defined(credit)) {
  231. throw new DeveloperError('credit must be defined');
  232. }
  233. //>>includeEnd('debug');
  234. var index;
  235. if (credit.hasImage()) {
  236. index = this._defaultImageCredits.indexOf(credit);
  237. if (index !== -1) {
  238. this._defaultImageCredits.splice(index, 1);
  239. }
  240. } else {
  241. index = this._defaultTextCredits.indexOf(credit);
  242. if (index !== -1) {
  243. this._defaultTextCredits.splice(index, 1);
  244. }
  245. }
  246. };
  247. /**
  248. * Resets the credit display to a beginning of frame state, clearing out current credits.
  249. *
  250. * @param {Credit} credit The credit to display
  251. */
  252. CreditDisplay.prototype.beginFrame = function() {
  253. this._currentFrameCredits.imageCredits.length = 0;
  254. this._currentFrameCredits.textCredits.length = 0;
  255. };
  256. /**
  257. * Sets the credit display to the end of frame state, displaying current credits in the credit container
  258. *
  259. * @param {Credit} credit The credit to display
  260. */
  261. CreditDisplay.prototype.endFrame = function() {
  262. var textCredits = this._defaultTextCredits.concat(this._currentFrameCredits.textCredits);
  263. var imageCredits = this._defaultImageCredits.concat(this._currentFrameCredits.imageCredits);
  264. displayTextCredits(this, textCredits);
  265. displayImageCredits(this, imageCredits);
  266. this._displayedCredits.textCredits = textCredits;
  267. this._displayedCredits.imageCredits = imageCredits;
  268. };
  269. /**
  270. * Destroys the resources held by this object. Destroying an object allows for deterministic
  271. * release of resources, instead of relying on the garbage collector to destroy this object.
  272. * <br /><br />
  273. * Once an object is destroyed, it should not be used; calling any function other than
  274. * <code>isDestroyed</code> will result in a {@link DeveloperError} exception. Therefore,
  275. * assign the return value (<code>undefined</code>) to the object as done in the example.
  276. *
  277. * @returns {undefined}
  278. *
  279. * @exception {DeveloperError} This object was destroyed, i.e., destroy() was called.
  280. */
  281. CreditDisplay.prototype.destroy = function() {
  282. this._container.removeChild(this._textContainer);
  283. this._container.removeChild(this._imageContainer);
  284. return destroyObject(this);
  285. };
  286. /**
  287. * Returns true if this object was destroyed; otherwise, false.
  288. * <br /><br />
  289. *
  290. * @returns {Boolean} <code>true</code> if this object was destroyed; otherwise, <code>false</code>.
  291. */
  292. CreditDisplay.prototype.isDestroyed = function() {
  293. return false;
  294. };
  295. return CreditDisplay;
  296. });