yet-another-github-card 1.0.1 → 1.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (71) hide show
  1. package/AUTHORS.md +3 -0
  2. package/CODE_OF_CONDUCT.md +3 -0
  3. package/CONTRIBUTING.md +5 -0
  4. package/LICENSE +21 -0
  5. package/README.md +4 -4
  6. package/SECURITY.md +13 -0
  7. package/css/theme-dark.css +10 -0
  8. package/css/theme-light.css +10 -0
  9. package/images/android-chrome-192x192.png +0 -0
  10. package/images/android-chrome-512x512.png +0 -0
  11. package/images/apple-touch-icon.png +0 -0
  12. package/images/blog.svg +4 -0
  13. package/images/commits.svg +44 -0
  14. package/images/contributions.svg +44 -0
  15. package/images/docs/favicon_io/android-chrome-192x192.png +0 -0
  16. package/images/docs/favicon_io/android-chrome-512x512.png +0 -0
  17. package/images/docs/favicon_io/apple-touch-icon.png +0 -0
  18. package/images/docs/favicon_io/favicon-16x16.png +0 -0
  19. package/images/docs/favicon_io/favicon-32x32.png +0 -0
  20. package/images/docs/favicon_io/favicon.ico +0 -0
  21. package/images/docs/favicon_io/site.webmanifest +1 -0
  22. package/images/docs/yagc-ico.png +0 -0
  23. package/images/docs/yagc-logo-square.png +0 -0
  24. package/images/docs/yagc-logo.png +0 -0
  25. package/images/docs/yagc-mkt.png +0 -0
  26. package/images/docs/yagc-showroom.kra +0 -0
  27. package/images/docs/yagc-showroom.png +0 -0
  28. package/images/docs/yagc-themes-style_cloud.png +0 -0
  29. package/images/docs/yagc-themes-style_default.png +0 -0
  30. package/images/docs/yagc-themes-style_polygon.png +0 -0
  31. package/images/favicon-16x16.png +0 -0
  32. package/images/favicon-32x32.png +0 -0
  33. package/images/favicon.ico +0 -0
  34. package/images/favicon_io/android-chrome-192x192.png +0 -0
  35. package/images/favicon_io/android-chrome-512x512.png +0 -0
  36. package/images/favicon_io/apple-touch-icon.png +0 -0
  37. package/images/favicon_io/favicon-16x16.png +0 -0
  38. package/images/favicon_io/favicon-32x32.png +0 -0
  39. package/images/favicon_io/favicon.ico +0 -0
  40. package/images/favicon_io/site.webmanifest +1 -0
  41. package/images/followers.svg +3 -0
  42. package/images/following.svg +40 -0
  43. package/images/generic.svg +4 -0
  44. package/images/gists.svg +38 -0
  45. package/images/github-logo.png +0 -0
  46. package/images/helm.svg +2 -0
  47. package/images/info-logo.svg +1 -0
  48. package/images/instagram.svg +54 -0
  49. package/images/linkedin.svg +3 -0
  50. package/images/npmjs.svg +44 -0
  51. package/images/nuget.svg +4 -0
  52. package/images/prs.svg +44 -0
  53. package/images/repositories.svg +38 -0
  54. package/images/rubygems.svg +4 -0
  55. package/images/site.webmanifest +1 -0
  56. package/images/sourceforge.svg +4 -0
  57. package/images/sponsor.svg +3 -0
  58. package/images/twitter.svg +3 -0
  59. package/images/yagc-ico.png +0 -0
  60. package/images/yagc-logo-square.png +0 -0
  61. package/images/yagc-logo.png +0 -0
  62. package/images/yagc-mkt.png +0 -0
  63. package/images/yagc-showroom.kra +0 -0
  64. package/images/yagc-showroom.png +0 -0
  65. package/images/yagc-themes-style_cloud.png +0 -0
  66. package/images/yagc-themes-style_default.png +0 -0
  67. package/images/yagc-themes-style_polygon.png +0 -0
  68. package/images/yarnpkg.svg +2 -0
  69. package/package.json +22 -4
  70. package/yagc.js +867 -0
  71. package/yagc.min.js +1 -0
package/yagc.js ADDED
@@ -0,0 +1,867 @@
1
+ import 'https://cdn.skypack.dev/graphql@15.8.0';
2
+ import 'https://cdn.skypack.dev/graphql-request@6.0.0';
3
+
4
+ // src/constants.ts
5
+
6
+ /**
7
+ * The base URL for the GraphQL API.
8
+ * @constant
9
+ */
10
+ const API_GQL_URL = 'https://yagc-api.notesoncloudcomputing.com/api/stats';
11
+
12
+ /**
13
+ * The base URL for the npm package.
14
+ * @constant
15
+ */
16
+ const NPM_URL = 'https://www.npmjs.com/package/yet-another-github-card';
17
+
18
+ /**
19
+ * The npm package name.
20
+ * @constant
21
+ */
22
+ const NPM_PKG = 'yet-another-github-card';
23
+
24
+ /**
25
+ * The npm package version.
26
+ * @constant
27
+ */
28
+ const NPM_VER = 'v1.0';
29
+
30
+ /**
31
+ * The default user profile data.
32
+ * @constant
33
+ */
34
+ const GH_DEF_COLLECTION = [{name: 'name'}, {name: 'bio'}, {name: 'createdAt'}, {name: 'location'}];
35
+
36
+ /**
37
+ * Theme Polygon style CSS Background-image code.
38
+ * @constant
39
+ */
40
+ const polygon = `
41
+ background-image: url("")
42
+ `;
43
+
44
+ /**
45
+ * Theme Cloud style CSS Background-image code.
46
+ * @constant
47
+ */
48
+ const cloud = `
49
+ background-image: url("")
50
+ `;
51
+
52
+ /**
53
+ * Default theme Circuit style CSS Background-image code.
54
+ * @constant
55
+ */
56
+ const defaultTheme = `
57
+ background-image: url("")
58
+ `;
59
+
60
+ // src/graphql.js
61
+
62
+ /**
63
+ * Description TODO
64
+ * @constant
65
+ */
66
+ const USERCARD = `
67
+ query getMyUsercard($login: String!) {
68
+ user(login: $login) {
69
+ createdAt
70
+ id
71
+ login
72
+ url
73
+ name
74
+ bio
75
+ avatarUrl
76
+ location
77
+ status {
78
+ message
79
+ }
80
+ websiteUrl
81
+ isGitHubStar
82
+ isHireable
83
+ socialAccounts (first: 10) {
84
+ nodes {
85
+ displayName
86
+ provider
87
+ url
88
+ }
89
+ }
90
+ starredRepositories (first: 10) {
91
+ totalCount
92
+ }
93
+ sponsors (first: 10) {
94
+ totalCount
95
+ }
96
+ }
97
+ }
98
+ `;
99
+
100
+ /**
101
+ * Description TODO
102
+ * @constant
103
+ */
104
+ const STARS = `
105
+ query getMyStars($login: String!) {
106
+ user(login: $login) {
107
+ repositories (first: 50, isFork: false, privacy: PUBLIC) {
108
+ totalCount
109
+ nodes {
110
+ stargazerCount
111
+ }
112
+ }
113
+ }
114
+ }
115
+ `;
116
+
117
+ /**
118
+ * Description TODO
119
+ * @constant
120
+ */
121
+ const GIST_STARS = `
122
+ query getMyGistsStars() {
123
+ viewer {
124
+ gists(first: 100, privacy: PUBLIC, orderBy: {field: CREATED_AT, direction: DESC}) {
125
+ nodes {
126
+ stargazerCount
127
+ }
128
+ }
129
+ }
130
+ }
131
+ `;
132
+
133
+ /**
134
+ * Description TODO
135
+ * @constant
136
+ */
137
+ const FOLLOWERS = `
138
+ query getMyFollowers($login: String!) {
139
+ user(login: $login) {
140
+ followers {
141
+ totalCount
142
+ }
143
+ }
144
+ }
145
+ `;
146
+
147
+ /**
148
+ * Description TODO
149
+ * @constant
150
+ */
151
+ const FOLLOWING = `
152
+ query getMyFollows($login: String!) {
153
+ user(login: $login) {
154
+ following {
155
+ totalCount
156
+ }
157
+ }
158
+ }
159
+ `;
160
+
161
+
162
+ /**
163
+ * Description TODO
164
+ * @constant
165
+ */
166
+ const COMMITS = `
167
+ query getMyCommits($login: String!) {
168
+ user(login: $login) {
169
+ repositories(first: 20, orderBy: {field: NAME, direction: ASC}) {
170
+ nodes {
171
+ defaultBranchRef {
172
+ target {
173
+ ... on Commit {
174
+ history(first: 10, author: {id: "MDQ6VXNlcjcxOTc3MzM"}) {
175
+ totalCount
176
+ }
177
+ }
178
+ }
179
+ }
180
+ }
181
+ pageInfo {
182
+ endCursor
183
+ startCursor
184
+ }
185
+ }
186
+ }
187
+ }
188
+ `;
189
+
190
+ /**
191
+ * Description TODO.
192
+ * @constant
193
+ */
194
+ const CONTRIBUTIONS = `
195
+ query getMyContributions($login: String!) {
196
+ user(login: $login) {
197
+ contributionsCollection {
198
+ contributionCalendar {
199
+ totalContributions
200
+ }
201
+ }
202
+ }
203
+ }
204
+ `;
205
+
206
+ /**
207
+ * Description TODO
208
+ * @constant
209
+ */
210
+ const PRS = `
211
+ query getMyPrs($login: String!) {
212
+ user(login: $login) {
213
+ pullRequests(first: 100, states: CLOSED) {
214
+ totalCount
215
+ }
216
+ }
217
+ }
218
+ `;
219
+
220
+ const yagcStyles = `.gh-card {
221
+ position: relative;
222
+ display: inline-block;
223
+ background: var(--body-bg);
224
+ border-radius: 1rem;
225
+ box-shadow: 0 12px 13px rgba(0,0,0,.16), 0 12px 13px rgba(0,0,0,.16);
226
+ font-family: helvetica, arial, 'system-ui';
227
+ text-align: center;
228
+ padding: 1em 1em;
229
+ margin: 1rem 1rem;
230
+ min-width: 300px;
231
+ transition: all .5s;
232
+ div, h1 {
233
+ margin: 0;
234
+ }
235
+ h1 {
236
+ font-size: 1em;
237
+ }
238
+ .location,
239
+ p {
240
+ color: var(--body-color);
241
+ font-size: 0.9em;
242
+ margin: auto;
243
+ }
244
+ a {
245
+ color: var(--body-color-a);
246
+ text-decoration: none;
247
+
248
+ &:hover {
249
+ filter: drop-shadow(3px 1px 3px #999)
250
+ }
251
+ }
252
+ a.created_at {
253
+ position: relative;
254
+ float: left;
255
+ z-index: 1;
256
+
257
+ img {
258
+ height: 2.5em;
259
+ width: auto;
260
+ }
261
+
262
+ img:hover {
263
+ filter: drop-shadow(3px 1px 3px #999)
264
+ }
265
+ }
266
+ #list-resources {
267
+ list-style: none;
268
+ text-align: left;
269
+ padding: 17px 9px;
270
+ position: absolute;
271
+ float: left;
272
+ z-index: 1;
273
+
274
+ li {
275
+ font-size: 0.8em;
276
+ color: #ffffff;
277
+ padding-bottom: 6px;
278
+ a {
279
+ color: #ffffff;
280
+ }
281
+ span {
282
+ margin-left: 5px;
283
+ vertical-align: super;
284
+ }
285
+ img:hover {
286
+ filter: drop-shadow(3px 1px 3px #999)
287
+ }
288
+ }
289
+ }
290
+ .since {
291
+ color: #FFF;
292
+ font-size: 0.8em;
293
+ margin-left: 3em;
294
+ line-height: 3em;
295
+ position: absolute;
296
+ left: 2em;
297
+ }
298
+
299
+ a.info {
300
+ position: relative;
301
+ float: right;
302
+ z-index: 1;
303
+
304
+ img {
305
+ height: 1.2em;
306
+ width: auto;
307
+ }
308
+
309
+ img:hover {
310
+ filter: drop-shadow(3px 1px 3px #999)
311
+ }
312
+ }
313
+ .cover {
314
+ height: 11rem;
315
+ width: 100%;
316
+ position: absolute;
317
+ left: 0;
318
+ top: 0;
319
+ background-position: bottom;
320
+ background-repeat: round;
321
+ border-radius: 1rem 1rem 0 0;
322
+ }
323
+ .name {
324
+ font-size: 1.5em;
325
+ }
326
+ .login {
327
+ color: var(--body-color);
328
+ margin-bottom: 1rem;
329
+ }
330
+ .avatar {
331
+ position: relative;
332
+ padding: 4rem 0 0 0;
333
+
334
+ img {
335
+ font-size: 2em;
336
+ height: auto;
337
+ width: 5em;
338
+ border-radius: 50%;
339
+ border: 3px solid var(--avatar-border-color);
340
+ filter: drop-shadow(0 0 10px var(--avatar-border-color));
341
+ }
342
+ }
343
+ .stars {
344
+ top: 10em;
345
+ position: absolute;
346
+ width: 4em;
347
+ height: 4em;
348
+ right: 15%;
349
+ display: inline;
350
+ background: no-repeat url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="256" height="256" viewBox="0 0 256 256" xml:space="preserve"><defs></defs><g style="stroke: none; stroke-width: 0; stroke-dasharray: none; stroke-linecap: butt; stroke-linejoin: miter; stroke-miterlimit: 10; fill: none; fill-rule: nonzero; opacity: 1;" transform="translate(1.4065934065934016 1.4065934065934016) scale(2.81 2.81)" ><path d="M 45 2.024 C 45 2.024 45 2.024 45 2.024 c -1.398 0 -2.649 0.778 -3.268 2.031 L 29.959 27.911 c -0.099 0.2 -0.29 0.338 -0.51 0.37 L 3.122 32.107 c -1.383 0.201 -2.509 1.151 -2.941 2.48 c -0.432 1.329 -0.079 2.76 0.922 3.736 l 19.049 18.569 c 0.16 0.156 0.233 0.38 0.195 0.599 L 15.85 83.71 c -0.236 1.377 0.319 2.743 1.449 3.564 c 1.129 0.821 2.6 0.927 3.839 0.279 l 23.547 -12.381 c 0.098 -0.051 0.206 -0.077 0.314 -0.077 C 51.721 53.905 50.301 28.878 45 2.024 z" style="stroke: none; stroke-width: 1; stroke-dasharray: none; stroke-linecap: butt; stroke-linejoin: miter; stroke-miterlimit: 10; fill: rgb(255,200,80); fill-rule: nonzero; opacity: 1;" transform=" matrix(1 0 0 1 0 0) " stroke-linecap="round" /><path d="M 45 2.024 C 45 2.024 45 2.024 45 2.024 c 1.398 0 2.649 0.778 3.268 2.031 l 11.773 23.856 c 0.099 0.2 0.29 0.338 0.51 0.37 l 26.326 3.826 c 1.383 0.201 2.509 1.151 2.941 2.48 c 0.432 1.329 0.079 2.76 -0.922 3.736 L 69.847 56.892 c -0.16 0.156 -0.233 0.38 -0.195 0.599 L 74.15 83.71 c 0.236 1.377 -0.319 2.743 -1.449 3.564 c -1.129 0.821 -2.6 0.927 -3.839 0.279 L 45.315 75.172 c -0.098 -0.051 -0.206 -0.077 -0.314 -0.077 C 37.08 54.593 38.849 29.395 45 2.024 z" style="stroke: none; stroke-width: 1; stroke-dasharray: none; stroke-linecap: butt; stroke-linejoin: miter; stroke-miterlimit: 10; fill: rgb(255,220,100); fill-rule: nonzero; opacity: 1;" transform=" matrix(1 0 0 1 0 0) " stroke-linecap="round" /></g></svg>');
351
+ background-size: 4em;
352
+ filter: drop-shadow(0 0 10px var(--avatar-border-color));
353
+ z-index: 1;
354
+
355
+ span {
356
+ color: var(--stars-color);
357
+ font-size: 1.2em;
358
+ font-weight: bold;
359
+ line-height: 4em;
360
+ text-align: center;
361
+ }
362
+ }
363
+ .footer {
364
+ border-top: 2px solid var(--stats-name-color);
365
+ padding: 1rem 0 0 0;
366
+
367
+ .item {
368
+ color: var(--body-color);
369
+ }
370
+ }
371
+ .grid-3 {
372
+ .item {
373
+ font-size: 1em;
374
+ width: 33%;
375
+ .total {
376
+ font-size: 2.4rem;
377
+ }
378
+ }
379
+ }
380
+ .grid-4 {
381
+ .item {
382
+ font-size: .8em;
383
+ width: 25%;
384
+ .total {
385
+ font-size: 2rem;
386
+ }
387
+ }
388
+ }
389
+ #list-items,
390
+ #list-social {
391
+ display: contents;
392
+ list-style: none;
393
+ padding: 0;
394
+
395
+ li {
396
+ display: inline-block;
397
+ span,
398
+ a {
399
+ font-size: 2em;
400
+ }
401
+ a:hover {
402
+ filter: drop-shadow(1px 1px 4px rgba(0, 0, 0, 1));
403
+ }
404
+ p {
405
+ font-size: 0.7em!important;
406
+ color: var(--stats-name-color);
407
+ }
408
+ }
409
+ li:not(:last-child) {
410
+ margin-right: 2rem;
411
+ }
412
+ }
413
+ }
414
+ #list-items {
415
+ li {
416
+ span {
417
+ color: var(--body-color);;
418
+ }
419
+ }
420
+ }
421
+ #list-social {
422
+ li {
423
+ img {
424
+ filter: var(--svg-color-a);
425
+ }
426
+ }
427
+ }
428
+ @media (min-width:400px) {
429
+ .gh-card {
430
+ .stars {
431
+ right: 25%;
432
+ }
433
+ }
434
+ }
435
+ @media (max-width:468px) {
436
+ .location {
437
+ display: none;
438
+ }
439
+ #list-social {
440
+ li {
441
+ margin-top: 10px;
442
+ }
443
+ }
444
+ }
445
+ @media (min-width:724px) {
446
+ .gh-card {
447
+ max-width: 1200px;
448
+ width: 90%;
449
+ .cover {
450
+ height: 15rem;
451
+ width: 100%;
452
+ }
453
+ .avatar {
454
+ position: relative;
455
+ padding: 4rem 0 0 0;
456
+
457
+ img {
458
+ width: 6em;
459
+ }
460
+ }
461
+ .stars {
462
+ top: 12em;
463
+ background-size: 4em;
464
+ right: 35%;
465
+ z-index: 1;
466
+ }
467
+ #list-social {
468
+ display: block;
469
+ }
470
+ }
471
+ }
472
+
473
+ @media (min-width:1024px) {
474
+ .gh-card {
475
+ .stars {
476
+ right: 39%;
477
+ }
478
+ }
479
+ }`;
480
+
481
+ /* eslint-disable no-restricted-properties */
482
+ function stringFormat(value) {
483
+ if (!(value >= 1000)) return value;
484
+ let newValue = value;
485
+ const suffixes = ['', 'k', 'm', 'b', 't'];
486
+ let suffixNum = 0;
487
+ while (newValue >= 1000) {
488
+ newValue /= 1000;
489
+ suffixNum += 1;
490
+ }
491
+
492
+ newValue = newValue.toPrecision(2);
493
+
494
+ newValue += suffixes[suffixNum];
495
+ return newValue;
496
+ }
497
+
498
+ let apiRequest, dataCollection, datasetLenght, dataName, dataHref, dataValue, responseLenght, shadowRoot, totalStars, userName, yagcTheme;
499
+
500
+ const template = document.createElement('template');
501
+
502
+ const queryMap = {
503
+ commits: COMMITS,
504
+ contributions: CONTRIBUTIONS,
505
+ following: FOLLOWING,
506
+ stars: STARS,
507
+ gist_stars: GIST_STARS,
508
+ prs: PRS,
509
+ usercard: USERCARD
510
+ };
511
+
512
+ const resourceMap = {
513
+ repositories: "https://github.com",
514
+ gists: "https://gist.github.com"
515
+ };
516
+
517
+ const themeMap = {
518
+ cloud: cloud,
519
+ polygon: polygon,
520
+ default: defaultTheme
521
+ };
522
+
523
+ const themeColor = {
524
+ cloud: `filter: invert(41%) sepia(9%) saturate(3861%) hue-rotate(174deg) brightness(92%) contrast(87%)`,
525
+ polygon: `filter: invert(33%) sepia(100%) saturate(785%) hue-rotate(193deg) brightness(97%) contrast(86%)`,
526
+ defaultTheme: `filter: invert(20%) sepia(35%) saturate(718%) hue-rotate(314deg) brightness(94%) contrast(95%)`
527
+ };
528
+
529
+ const faviconMap = {
530
+ generic: `./images/generic.svg`,
531
+ github: `./images/github.svg`,
532
+ helm: `./images/helm.svg`,
533
+ instagram: `./images/instagram.svg`,
534
+ linkedin: `./images/linkedin.svg`,
535
+ npmjs: `./images/npmjs.svg`,
536
+ nuget: `./images/nuget.svg`,
537
+ rubygems: `./images/rubygems.svg`,
538
+ sourceforge: `./images/sourceforge.svg`,
539
+ twitter: `./images/twitter.svg`,
540
+ yarnpkg: `./images/yarnpkg.svg`,
541
+ default: `./images/generic.svg`
542
+ };
543
+
544
+
545
+ template.innerHTML = /*html*/ `
546
+ <!--
547
+ Created using npm package ${NPM_PKG} ${NPM_VER}
548
+ ${NPM_URL}
549
+ -->
550
+ <style>${yagcStyles}</style>
551
+ <div class="gh-card">
552
+ <div class="cover" style=""></div>
553
+ <div class="wrapper">
554
+ <a href="" target="_blank" class="created_at" rel="noopener"><img src="./images/github-logo.png" border="0" alt="Account since"></a><span class="since">Since <slot name="createdAt">{createdAt}</slot></span>
555
+ <a href="${NPM_URL}" target="_blank" class="info" rel="noopener"><img src="./images/info-logo.svg" border="0" alt="${NPM_PKG}"></a><br>
556
+ <ul id="list-resources"></ul>
557
+ <div class="avatar">
558
+ <img src=""/></slot>
559
+ <div class="stars"><span><slot name="stars">{stars}</span></div>
560
+ </div>
561
+ <h1><a class="name" href="" target="_blank" rel="noopener"><slot name="name">{name}</slot></a></h1>
562
+ <div class="location"><slot name="location">{location}</slot></div>
563
+ <p><slot name="bio">{bio}</slot></p>
564
+ <div class="grid-4">
565
+ <ul id="list-social"></ul>
566
+ </div>
567
+ <div class="footer grid-4">
568
+ <ul id="list-items"></ul>
569
+ </div>
570
+ </div>
571
+ </div>
572
+ `;
573
+
574
+ class githubCard extends HTMLElement {
575
+ static get observedAttributes() {
576
+ return ["data-theme", "data-style"];
577
+ }
578
+
579
+ constructor() {
580
+ super();
581
+ const shadow = this.attachShadow({ mode: "open" });
582
+ shadow.appendChild(template.content.cloneNode(true));
583
+ }
584
+
585
+
586
+ connectedCallback() {
587
+ shadowRoot = document.querySelector('github-card').shadowRoot;
588
+
589
+ apiRequest = queryMap['usercard'];
590
+ userName = this.getAttribute('data-user');
591
+ yagcTheme = this.getAttribute('data-style').toString();
592
+
593
+ this.changeStyle(yagcTheme);
594
+
595
+ this.main(apiRequest)
596
+ .then((response) => {
597
+ responseLenght = Object.keys(response.user).length;
598
+ datasetLenght = Object.keys(GH_DEF_COLLECTION).length;
599
+
600
+ for (let i = 0; i < responseLenght; i++) {
601
+ if(response.user.hasOwnProperty(Object.values(GH_DEF_COLLECTION[i]|| {}))) {
602
+ let value = Object.values(GH_DEF_COLLECTION[i]);
603
+
604
+ if (Object.values(GH_DEF_COLLECTION[i]) == "createdAt") {
605
+ shadowRoot.querySelector(`slot[name=${value}`).textContent = response.user[value].substr(0,4);
606
+
607
+ } else {
608
+ shadowRoot.querySelector(`slot[name=${value}`).textContent = response.user[value];
609
+ }
610
+ }
611
+ }
612
+
613
+ shadowRoot.querySelector(".avatar img").setAttribute("src", response.user["avatarUrl"]);
614
+ shadowRoot.querySelector(".gh-card a.created_at").setAttribute("href", response.user["url"]);
615
+ shadowRoot.querySelector(".name").setAttribute("href", response.user["url"]);
616
+
617
+ if (response.user["websiteUrl"]) { this.createListElements("list-social", response.user["websiteUrl"], "blog", null, "link", 30); }
618
+
619
+ if (response.user["socialAccounts"]) {
620
+ responseLenght = Object.keys(response.user.socialAccounts.nodes).length;
621
+
622
+ for (let i = 0; i < responseLenght; i++) {
623
+ this.createListElements("list-social", response.user.socialAccounts.nodes[i].url, response.user.socialAccounts.nodes[i].provider.toLowerCase(), null, "link", 30);
624
+ }
625
+ }
626
+ })
627
+
628
+ .catch((error) => {
629
+ console.log(error);
630
+ });
631
+
632
+ dataCollection = this.getAttribute("data-user-stars").toString();
633
+ if (dataCollection == "true") {
634
+
635
+ this.main(STARS)
636
+ .then((response) => {
637
+
638
+ totalStars = 0;
639
+
640
+ Object.keys(response.user).forEach(key => {
641
+ const childKey = response.user[key];
642
+ this.createListElements("list-resources", `${resourceMap[key]}/${userName}`, key, childKey["totalCount"] + " " + key, "link", 20);
643
+ Object.keys(childKey).forEach(key => {
644
+ const childDescKey = childKey[key];
645
+ Object.keys(childDescKey).forEach(key => {
646
+ totalStars = totalStars += childDescKey[key].stargazerCount;
647
+ });
648
+ });
649
+ });
650
+ shadowRoot.querySelector(`slot[name=stars`).textContent = stringFormat(totalStars);
651
+ })
652
+
653
+ .catch((error) => {
654
+ console.log(error);
655
+ });
656
+ }
657
+
658
+ dataCollection = this.getAttribute("data-user-stats");
659
+ const userCollection = JSON.parse(dataCollection);
660
+ datasetLenght = Object.keys(userCollection).length;
661
+
662
+ for (let i = 0; i < datasetLenght; i++) {
663
+
664
+ apiRequest = queryMap[userCollection[i].name] || FOLLOWERS;
665
+
666
+ this.main(apiRequest)
667
+ .then((response) => {
668
+ dataHref = '';
669
+ dataName = userCollection[i].name;
670
+ dataValue = 0;
671
+ dataValue = this.getStats(response);
672
+ this.createListElements("list-items", dataHref, dataName.toUpperCase(), stringFormat(dataValue), "text", 25);
673
+ })
674
+
675
+ .catch((error) => {
676
+ //console.log(error);
677
+ });
678
+ }
679
+ }
680
+
681
+
682
+ getStats(obj) {
683
+ const processProperty = (property) => {
684
+ if (property.totalCount || property.totalContributions) {
685
+ dataValue += property.totalCount || property.totalContributions;
686
+ } else if (property instanceof Object) {
687
+ Object.keys(property).forEach((key) => {
688
+ processProperty(property[key]);
689
+ });
690
+ }
691
+ };
692
+
693
+ processProperty(obj); // Start processing from the top-level object
694
+ return dataValue;
695
+ }
696
+
697
+
698
+ getRootDomain(url) {
699
+ if (url) {
700
+ const parsedUrl = new URL(url);
701
+ const hostname = parsedUrl.hostname;
702
+ const hostnameParts = hostname.split('.');
703
+
704
+ if (hostnameParts.length < 2) {
705
+ return hostname; // Return the whole hostname if no subdomain
706
+ }
707
+
708
+ if (hostnameParts.length === 3 && hostnameParts[0] === 'www') {
709
+ return hostnameParts[1]; // Root domain is the part before "www"
710
+ }
711
+ // Loop through known domains to check for a match
712
+ for (const domain of Object.keys(faviconMap)) {
713
+ if (hostnameParts.includes(domain)) {
714
+ // If a match is found, return the root domain based on its position
715
+ const rootDomainIndex = hostnameParts.indexOf(domain);
716
+ return hostnameParts[rootDomainIndex - 1];
717
+ } else {
718
+ return hostnameParts[hostnameParts.length - 2];
719
+ }
720
+ }
721
+ return hostnameParts[hostnameParts.length - 2]; // Default root domain (second-to-last element)
722
+ }
723
+ }
724
+
725
+
726
+ async createListElements(parentId, href, name, value = 0, type, size = 16) {
727
+ const shadowRoot = document.querySelector('github-card').shadowRoot;
728
+ const itemNode = document.createElement('li');
729
+ itemNode.classList.add('list-item');
730
+
731
+ let srcImg, listImg, listEle, listChild;
732
+ if (['gists', 'repositories'].includes(name)) {
733
+ srcImg = `./images/${name}.svg`;
734
+ href = href + '?tab=repositories';
735
+ } else if (name === 'blog') {
736
+ const validUrl = await this.urlCheckStatus(`${href}/assets/images/favicon_io/favicon.svg`);
737
+ if (validUrl) {
738
+ srcImg = validUrl;
739
+ } else {
740
+ srcImg = `./images/generic.svg`; // Fallback if URL check fails
741
+ }
742
+ } else {
743
+ const domain = this.getRootDomain(href);
744
+ srcImg = faviconMap[domain] || `./images/generic.svg`; // Default to generic if not found
745
+ }
746
+
747
+ switch (parentId) {
748
+ case 'list-resources':
749
+ case 'list-social':
750
+ listEle = document.createElement('a');
751
+ listEle.textContent = value;
752
+ listEle.setAttribute('alt', name);
753
+ listEle.setAttribute('href', href);
754
+ listEle.setAttribute('target', '_blank');
755
+ listEle.setAttribute('rel', 'noopener');
756
+ listEle.setAttribute('title', href);
757
+ listChild = null;
758
+ break;
759
+ default:
760
+ listEle = document.createElement('span');
761
+ listEle.textContent = value;
762
+ listChild = document.createElement('p');
763
+ listChild.textContent = name;
764
+ itemNode.appendChild(listChild);
765
+ srcImg = `./images/${name.toLowerCase()}.svg`;
766
+ }
767
+
768
+ listImg = document.createElement('img');
769
+ Object.assign(listImg, {
770
+ src: `${srcImg}`,
771
+ height: size,
772
+ width: size,
773
+ alt: name,
774
+ class: themeColor[yagcTheme]
775
+ });
776
+
777
+ listEle.prepend(listImg);
778
+ itemNode.appendChild(listEle);
779
+ if (listChild != null) { itemNode.appendChild(listChild); }
780
+
781
+ const parentElement = shadowRoot.getElementById(parentId);
782
+ if (parentElement) {
783
+ parentElement.appendChild(itemNode);
784
+ } else {
785
+ console.error(`Parent element with id "${parentId}" not found.`);
786
+ }
787
+ }
788
+
789
+
790
+ changeStyle(yagcTheme) {
791
+ const shadowRoot = document.querySelector('github-card').shadowRoot;
792
+ const cover = shadowRoot.querySelector(".cover");
793
+ cover.setAttribute("style", themeMap[yagcTheme] || defaultTheme);
794
+ this.changeStyleColor(yagcTheme);
795
+ }
796
+
797
+
798
+ changeStyleColor(yagcTheme) {
799
+ setTimeout(() => {
800
+ const wrapper = shadowRoot.querySelector("#list-items").querySelectorAll(".list-item");
801
+ const elementsArray = Array.from(wrapper);
802
+
803
+ elementsArray.forEach(listItem => {
804
+ const imgElements = listItem.querySelectorAll("img");
805
+ imgElements.forEach(imgEle => {
806
+ imgEle.setAttribute("style", themeColor[yagcTheme]);
807
+ });
808
+ });
809
+ }, 2000);
810
+ }
811
+
812
+
813
+ attributeChangedCallback(name, oldValue, newValue) {
814
+ console.log(`${name}'s value has been changed from ${oldValue} to ${newValue}`);
815
+ this.changeStyle(newValue);
816
+ }
817
+
818
+
819
+ urlCheckStatus(url) {
820
+ return fetch(url)
821
+ .then((response) => {
822
+ if (response.status == 200) {
823
+ return url;
824
+ } else {
825
+ return null;
826
+ }
827
+ })
828
+ .catch((error) => {
829
+ console.log(error);
830
+ });
831
+ }
832
+
833
+
834
+ async main(graphQLQuery) {
835
+
836
+ const endPoint = API_GQL_URL;
837
+ const variables = {
838
+ login: `${userName}`,
839
+ };
840
+
841
+ try {
842
+ const response = await fetch(endPoint, {
843
+ method: 'POST',
844
+ headers: {
845
+ 'Content-Type': 'application/json',
846
+ },
847
+ body: JSON.stringify({
848
+ query: graphQLQuery,
849
+ variables: variables,
850
+ }),
851
+ });
852
+
853
+ if (!response.ok) {
854
+ throw new Error(`Error en el proxy: ${response.statusText}`);
855
+ }
856
+
857
+ const data = await response.json();
858
+ return data;
859
+
860
+ } catch (error) {
861
+ console.error("Error al obtener datos del backend proxy:", error);
862
+ throw error;
863
+ }
864
+ }
865
+ }
866
+
867
+ customElements.define('github-card', githubCard);