├── .gitignore
├── .travis.yml
├── LICENSE
├── README.md
├── docs
├── nimdoc.out.css
└── stacks.html
├── src
└── stacks.nim
├── stacks.nimble
└── tests
├── config.nims
└── test_stacks.nim
/.gitignore:
--------------------------------------------------------------------------------
1 | nimcache/
2 | .DS_Store
3 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | sudo: required
2 | services:
3 | - docker
4 | before_install:
5 | - docker pull nimlang/nim
6 | script:
7 | - docker run nimlang/nim nim --version
8 | - docker run -v "$(pwd):/project" -w /project nimlang/nim sh -c "nimble install -dy && nimble test"
9 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2019 Max Skybin
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # nim-stacks
2 | Pure Nim stack implementation based on sequences
3 |
4 | * [Full documentation](https://rustomax.github.io/dev/nim/stacks/stacks.html)
5 |
6 | Version 0.4.2
7 |
8 | ### Installation
9 | ```sh
10 | nimble install stacks
11 | ```
12 |
13 | ### Example
14 |
15 | ```nim
16 | import stacks
17 |
18 | proc isPaired*(s: string): bool =
19 | ## Algorithm to detect unbalanced brackets using a stack
20 | var stack = Stack[char]()
21 |
22 | for c in s:
23 | case c:
24 | of '{': stack.push('}')
25 | of '[': stack.push(']')
26 | of '(': stack.push(')')
27 | of '}', ']', ')':
28 | if stack.isEmpty or stack.pop() != c: return false
29 | else: discard
30 |
31 | stack.isEmpty()
32 |
33 | when isMainModule:
34 | assert isPaired("(((185 + 223.85) * 15) - 543)/2") == true
35 | assert isPaired("for (i = 1; i < 11; ++i)\n{printf(\"i\");}\nreturn 0;}\n}") == false
36 | ```
37 |
38 |
--------------------------------------------------------------------------------
/docs/nimdoc.out.css:
--------------------------------------------------------------------------------
1 | /*
2 | Stylesheet for use with Docutils/rst2html.
3 |
4 | See http://docutils.sf.net/docs/howto/html-stylesheets.html for how to
5 | customize this style sheet.
6 |
7 | Modified from Chad Skeeters' rst2html-style
8 | https://bitbucket.org/cskeeters/rst2html-style/
9 |
10 | Modified by Boyd Greenfield and narimiran
11 | */
12 |
13 | :root {
14 | --primary-background: #fff;
15 | --secondary-background: ghostwhite;
16 | --third-background: #e8e8e8;
17 | --border: #dde;
18 | --text: #222;
19 | --anchor: #07b;
20 | --anchor-focus: #607c9f;
21 | --input-focus: #1fa0eb;
22 | --strong: #3c3c3c;
23 | --hint: #9A9A9A;
24 | --nim-sprite-base64: url("");
25 |
26 | --keyword: #5e8f60;
27 | --identifier: #222;
28 | --comment: #484a86;
29 | --operator: #155da4;
30 | --punctuation: black;
31 | --other: black;
32 | --escapeSequence: #c4891b;
33 | --number: #252dbe;
34 | --literal: #a4255b;
35 | --raw-data: #a4255b;
36 | }
37 |
38 | [data-theme="dark"] {
39 | --primary-background: #171921;
40 | --secondary-background: #1e202a;
41 | --third-background: #2b2e3b;
42 | --border: #0e1014;
43 | --text: #fff;
44 | --anchor: #8be9fd;
45 | --anchor-focus: #8be9fd;
46 | --input-focus: #8be9fd;
47 | --strong: #bd93f9;
48 | --hint: #7A7C85;
49 | --nim-sprite-base64: url("");
50 |
51 | --keyword: #ff79c6;
52 | --identifier: #f8f8f2;
53 | --comment: #6272a4;
54 | --operator: #ff79c6;
55 | --punctuation: #f8f8f2;
56 | --other: #f8f8f2;
57 | --escapeSequence: #bd93f9;
58 | --number: #bd93f9;
59 | --literal: #f1fa8c;
60 | --raw-data: #8be9fd;
61 | }
62 |
63 | .theme-switch-wrapper {
64 | display: flex;
65 | align-items: center;
66 | }
67 |
68 | .theme-switch-wrapper em {
69 | margin-left: 10px;
70 | font-size: 1rem;
71 | }
72 |
73 | .theme-switch {
74 | display: inline-block;
75 | height: 22px;
76 | position: relative;
77 | width: 50px;
78 | }
79 |
80 | .theme-switch input {
81 | display: none;
82 | }
83 |
84 | .slider {
85 | background-color: #ccc;
86 | bottom: 0;
87 | cursor: pointer;
88 | left: 0;
89 | position: absolute;
90 | right: 0;
91 | top: 0;
92 | transition: .4s;
93 | }
94 |
95 | .slider:before {
96 | background-color: #fff;
97 | bottom: 4px;
98 | content: "";
99 | height: 13px;
100 | left: 4px;
101 | position: absolute;
102 | transition: .4s;
103 | width: 13px;
104 | }
105 |
106 | input:checked + .slider {
107 | background-color: #66bb6a;
108 | }
109 |
110 | input:checked + .slider:before {
111 | transform: translateX(26px);
112 | }
113 |
114 | .slider.round {
115 | border-radius: 17px;
116 | }
117 |
118 | .slider.round:before {
119 | border-radius: 50%;
120 | }
121 |
122 | html {
123 | font-size: 100%;
124 | -webkit-text-size-adjust: 100%;
125 | -ms-text-size-adjust: 100%; }
126 |
127 | body {
128 | font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif;
129 | font-weight: 400;
130 | font-size: 1.125em;
131 | line-height: 1.5;
132 | color: var(--text);
133 | background-color: var(--primary-background); }
134 |
135 | /* Skeleton grid */
136 | .container {
137 | position: relative;
138 | width: 100%;
139 | max-width: 1050px;
140 | margin: 0 auto;
141 | padding: 0;
142 | box-sizing: border-box; }
143 |
144 | .column,
145 | .columns {
146 | width: 100%;
147 | float: left;
148 | box-sizing: border-box;
149 | margin-left: 1%;
150 | }
151 |
152 | .column:first-child,
153 | .columns:first-child {
154 | margin-left: 0; }
155 |
156 | .three.columns {
157 | width: 22%;
158 | line-break: anywhere;
159 | }
160 |
161 | .nine.columns {
162 | width: 77.0%; }
163 |
164 | .twelve.columns {
165 | width: 100%;
166 | margin-left: 0; }
167 |
168 | @media screen and (max-width: 860px) {
169 | .three.columns {
170 | display: none;
171 | }
172 | .nine.columns {
173 | width: 98.0%;
174 | }
175 | body {
176 | font-size: 1em;
177 | line-height: 1.35;
178 | }
179 | }
180 |
181 | cite {
182 | font-style: italic !important; }
183 |
184 |
185 | /* Nim search input */
186 | div#searchInputDiv {
187 | margin-bottom: 1em;
188 | }
189 | input#searchInput {
190 | width: 80%;
191 | }
192 |
193 | /*
194 | * Some custom formatting for input forms.
195 | * This also fixes input form colors on Firefox with a dark system theme on Linux.
196 | */
197 | input {
198 | -moz-appearance: none;
199 | background-color: var(--secondary-background);
200 | color: var(--text);
201 | border: 1px solid var(--border);
202 | font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif;
203 | font-size: 0.9em;
204 | padding: 6px;
205 | }
206 |
207 | input:focus {
208 | border: 1px solid var(--input-focus);
209 | box-shadow: 0 0 3px var(--input-focus);
210 | }
211 |
212 | select {
213 | -moz-appearance: none;
214 | background-color: var(--secondary-background);
215 | color: var(--text);
216 | border: 1px solid var(--border);
217 | font-family: "Lato", "Helvetica Neue", "HelveticaNeue", Helvetica, Arial, sans-serif;
218 | font-size: 0.9em;
219 | padding: 6px;
220 | }
221 |
222 | select:focus {
223 | border: 1px solid var(--input-focus);
224 | box-shadow: 0 0 3px var(--input-focus);
225 | }
226 |
227 | /* Docgen styles */
228 | /* Links */
229 | a {
230 | color: var(--anchor);
231 | text-decoration: none;
232 | }
233 |
234 | a span.Identifier {
235 | text-decoration: underline;
236 | text-decoration-color: #aab;
237 | }
238 |
239 | a.reference-toplevel {
240 | font-weight: bold;
241 | }
242 |
243 | a.toc-backref {
244 | text-decoration: none;
245 | color: var(--text); }
246 |
247 | a.link-seesrc {
248 | color: #607c9f;
249 | font-size: 0.9em;
250 | font-style: italic; }
251 |
252 | a:hover,
253 | a:focus {
254 | color: var(--anchor-focus);
255 | text-decoration: underline; }
256 |
257 | a:hover span.Identifier {
258 | color: var(--anchor);
259 | }
260 |
261 |
262 | sub,
263 | sup {
264 | position: relative;
265 | font-size: 75%;
266 | line-height: 0;
267 | vertical-align: baseline; }
268 |
269 | sup {
270 | top: -0.5em; }
271 |
272 | sub {
273 | bottom: -0.25em; }
274 |
275 | img {
276 | width: auto;
277 | height: auto;
278 | max-width: 100%;
279 | vertical-align: middle;
280 | border: 0;
281 | -ms-interpolation-mode: bicubic; }
282 |
283 | @media print {
284 | * {
285 | color: black !important;
286 | text-shadow: none !important;
287 | background: transparent !important;
288 | box-shadow: none !important; }
289 |
290 | a,
291 | a:visited {
292 | text-decoration: underline; }
293 |
294 | a[href]:after {
295 | content: " (" attr(href) ")"; }
296 |
297 | abbr[title]:after {
298 | content: " (" attr(title) ")"; }
299 |
300 | .ir a:after,
301 | a[href^="javascript:"]:after,
302 | a[href^="#"]:after {
303 | content: ""; }
304 |
305 | pre,
306 | blockquote {
307 | border: 1px solid #999;
308 | page-break-inside: avoid; }
309 |
310 | thead {
311 | display: table-header-group; }
312 |
313 | tr,
314 | img {
315 | page-break-inside: avoid; }
316 |
317 | img {
318 | max-width: 100% !important; }
319 |
320 | @page {
321 | margin: 0.5cm; }
322 |
323 | h1 {
324 | page-break-before: always; }
325 |
326 | h1.title {
327 | page-break-before: avoid; }
328 |
329 | p,
330 | h2,
331 | h3 {
332 | orphans: 3;
333 | widows: 3; }
334 |
335 | h2,
336 | h3 {
337 | page-break-after: avoid; }
338 | }
339 |
340 |
341 | p {
342 | margin-top: 0.5em;
343 | margin-bottom: 0.5em;
344 | }
345 |
346 | small {
347 | font-size: 85%; }
348 |
349 | strong {
350 | font-weight: 600;
351 | font-size: 0.95em;
352 | color: var(--strong);
353 | }
354 |
355 | em {
356 | font-style: italic; }
357 |
358 | h1 {
359 | font-size: 1.8em;
360 | font-weight: 400;
361 | padding-bottom: .25em;
362 | border-bottom: 6px solid var(--third-background);
363 | margin-top: 2.5em;
364 | margin-bottom: 1em;
365 | line-height: 1.2em; }
366 |
367 | h1.title {
368 | padding-bottom: 1em;
369 | border-bottom: 0px;
370 | font-size: 2.5em;
371 | text-align: center;
372 | font-weight: 900;
373 | margin-top: 0.75em;
374 | margin-bottom: 0em;
375 | }
376 |
377 | h2 {
378 | font-size: 1.3em;
379 | margin-top: 2em; }
380 |
381 | h2.subtitle {
382 | text-align: center; }
383 |
384 | h3 {
385 | font-size: 1.125em;
386 | font-style: italic;
387 | margin-top: 1.5em; }
388 |
389 | h4 {
390 | font-size: 1.125em;
391 | margin-top: 1em; }
392 |
393 | h5 {
394 | font-size: 1.125em;
395 | margin-top: 0.75em; }
396 |
397 | h6 {
398 | font-size: 1.1em; }
399 |
400 |
401 | ul,
402 | ol {
403 | padding: 0;
404 | margin-top: 0.5em;
405 | margin-left: 0.75em; }
406 |
407 | ul ul,
408 | ul ol,
409 | ol ol,
410 | ol ul {
411 | margin-bottom: 0;
412 | margin-left: 1.25em; }
413 |
414 | li {
415 | list-style-type: circle;
416 | }
417 |
418 | ul.simple-boot li {
419 | list-style-type: none;
420 | margin-left: 0em;
421 | margin-bottom: 0.5em;
422 | }
423 |
424 | ol.simple > li, ul.simple > li {
425 | margin-bottom: 0.2em;
426 | margin-left: 0.4em }
427 |
428 | ul.simple.simple-toc > li {
429 | margin-top: 1em;
430 | }
431 |
432 | ul.simple-toc {
433 | list-style: none;
434 | font-size: 0.9em;
435 | margin-left: -0.3em;
436 | margin-top: 1em; }
437 |
438 | ul.simple-toc > li {
439 | list-style-type: none;
440 | }
441 |
442 | ul.simple-toc-section {
443 | list-style-type: circle;
444 | margin-left: 0.8em;
445 | color: #6c9aae; }
446 |
447 | ul.nested-toc-section {
448 | list-style-type: circle;
449 | margin-left: -0.75em;
450 | color: var(--text);
451 | }
452 |
453 | ul.nested-toc-section > li {
454 | margin-left: 1.25em;
455 | }
456 |
457 |
458 | ol.arabic {
459 | list-style: decimal; }
460 |
461 | ol.loweralpha {
462 | list-style: lower-alpha; }
463 |
464 | ol.upperalpha {
465 | list-style: upper-alpha; }
466 |
467 | ol.lowerroman {
468 | list-style: lower-roman; }
469 |
470 | ol.upperroman {
471 | list-style: upper-roman; }
472 |
473 | ul.auto-toc {
474 | list-style-type: none; }
475 |
476 |
477 | dl {
478 | margin-bottom: 1.5em; }
479 |
480 | dt {
481 | margin-bottom: -0.5em;
482 | margin-left: 0.0em; }
483 |
484 | dd {
485 | margin-left: 2.0em;
486 | margin-bottom: 3.0em;
487 | margin-top: 0.5em; }
488 |
489 |
490 | hr {
491 | margin: 2em 0;
492 | border: 0;
493 | border-top: 1px solid #aaa; }
494 |
495 | blockquote {
496 | font-size: 0.9em;
497 | font-style: italic;
498 | padding-left: 0.5em;
499 | margin-left: 0;
500 | border-left: 5px solid #bbc;
501 | }
502 |
503 | .pre {
504 | font-family: "Source Code Pro", Monaco, Menlo, Consolas, "Courier New", monospace;
505 | font-weight: 500;
506 | font-size: 0.85em;
507 | color: var(--text);
508 | background-color: var(--third-background);
509 | padding-left: 3px;
510 | padding-right: 3px;
511 | border-radius: 4px;
512 | }
513 |
514 | pre {
515 | font-family: "Source Code Pro", Monaco, Menlo, Consolas, "Courier New", monospace;
516 | color: var(--text);
517 | font-weight: 500;
518 | display: inline-block;
519 | box-sizing: border-box;
520 | min-width: 100%;
521 | padding: 0.5em;
522 | margin-top: 0.5em;
523 | margin-bottom: 0.5em;
524 | font-size: 0.85em;
525 | white-space: pre !important;
526 | overflow-y: hidden;
527 | overflow-x: visible;
528 | background-color: var(--secondary-background);
529 | border: 1px solid var(--border);
530 | -webkit-border-radius: 6px;
531 | -moz-border-radius: 6px;
532 | border-radius: 6px; }
533 |
534 | .pre-scrollable {
535 | max-height: 340px;
536 | overflow-y: scroll; }
537 |
538 |
539 | /* Nim line-numbered tables */
540 | .line-nums-table {
541 | width: 100%;
542 | table-layout: fixed; }
543 |
544 | table.line-nums-table {
545 | border-radius: 4px;
546 | border: 1px solid #cccccc;
547 | background-color: ghostwhite;
548 | border-collapse: separate;
549 | margin-top: 15px;
550 | margin-bottom: 25px; }
551 |
552 | .line-nums-table tbody {
553 | border: none; }
554 |
555 | .line-nums-table td pre {
556 | border: none;
557 | background-color: transparent; }
558 |
559 | .line-nums-table td.blob-line-nums {
560 | width: 28px; }
561 |
562 | .line-nums-table td.blob-line-nums pre {
563 | color: #b0b0b0;
564 | -webkit-filter: opacity(75%);
565 | text-align: right;
566 | border-color: transparent;
567 | background-color: transparent;
568 | padding-left: 0px;
569 | margin-left: 0px;
570 | padding-right: 0px;
571 | margin-right: 0px; }
572 |
573 |
574 | table {
575 | max-width: 100%;
576 | background-color: transparent;
577 | margin-top: 0.5em;
578 | margin-bottom: 1.5em;
579 | border-collapse: collapse;
580 | border-color: var(--third-background);
581 | border-spacing: 0;
582 | font-size: 0.9em;
583 | }
584 |
585 | table th, table td {
586 | padding: 0px 0.5em 0px;
587 | border-color: var(--third-background);
588 | }
589 |
590 | table th {
591 | background-color: var(--third-background);
592 | border-color: var(--third-background);
593 | font-weight: bold; }
594 |
595 | table th.docinfo-name {
596 | background-color: transparent;
597 | text-align: right;
598 | }
599 |
600 | table tr:hover {
601 | background-color: var(--third-background); }
602 |
603 |
604 | /* rst2html default used to remove borders from tables and images */
605 | .borderless, table.borderless td, table.borderless th {
606 | border: 0; }
607 |
608 | table.borderless td, table.borderless th {
609 | /* Override padding for "table.docutils td" with "! important".
610 | The right padding separates the table cells. */
611 | padding: 0 0.5em 0 0 !important; }
612 |
613 | .first {
614 | /* Override more specific margin styles with "! important". */
615 | margin-top: 0 !important; }
616 |
617 | .last, .with-subtitle {
618 | margin-bottom: 0 !important; }
619 |
620 | .hidden {
621 | display: none; }
622 |
623 | blockquote.epigraph {
624 | margin: 2em 5em; }
625 |
626 | dl.docutils dd {
627 | margin-bottom: 0.5em; }
628 |
629 | object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
630 | overflow: hidden; }
631 |
632 |
633 | div.figure {
634 | margin-left: 2em;
635 | margin-right: 2em; }
636 |
637 | div.footer, div.header {
638 | clear: both;
639 | text-align: center;
640 | color: #666;
641 | font-size: smaller; }
642 |
643 | div.footer {
644 | padding-top: 5em;
645 | }
646 |
647 | div.line-block {
648 | display: block;
649 | margin-top: 1em;
650 | margin-bottom: 1em; }
651 |
652 | div.line-block div.line-block {
653 | margin-top: 0;
654 | margin-bottom: 0;
655 | margin-left: 1.5em; }
656 |
657 | div.topic {
658 | margin: 2em; }
659 |
660 | div.search_results {
661 | background-color: var(--third-background);
662 | margin: 3em;
663 | padding: 1em;
664 | border: 1px solid #4d4d4d;
665 | }
666 |
667 | div#global-links ul {
668 | margin-left: 0;
669 | list-style-type: none;
670 | }
671 |
672 | div#global-links > simple-boot {
673 | margin-left: 3em;
674 | }
675 |
676 | hr.docutils {
677 | width: 75%; }
678 |
679 | img.align-left, .figure.align-left, object.align-left {
680 | clear: left;
681 | float: left;
682 | margin-right: 1em; }
683 |
684 | img.align-right, .figure.align-right, object.align-right {
685 | clear: right;
686 | float: right;
687 | margin-left: 1em; }
688 |
689 | img.align-center, .figure.align-center, object.align-center {
690 | display: block;
691 | margin-left: auto;
692 | margin-right: auto; }
693 |
694 | .align-left {
695 | text-align: left; }
696 |
697 | .align-center {
698 | clear: both;
699 | text-align: center; }
700 |
701 | .align-right {
702 | text-align: right; }
703 |
704 | /* reset inner alignment in figures */
705 | div.align-right {
706 | text-align: inherit; }
707 |
708 | p.attribution {
709 | text-align: right;
710 | margin-left: 50%; }
711 |
712 | p.caption {
713 | font-style: italic; }
714 |
715 | p.credits {
716 | font-style: italic;
717 | font-size: smaller; }
718 |
719 | p.label {
720 | white-space: nowrap; }
721 |
722 | p.rubric {
723 | font-weight: bold;
724 | font-size: larger;
725 | color: maroon;
726 | text-align: center; }
727 |
728 | p.topic-title {
729 | font-weight: bold; }
730 |
731 | pre.address {
732 | margin-bottom: 0;
733 | margin-top: 0;
734 | font: inherit; }
735 |
736 | pre.literal-block, pre.doctest-block, pre.math, pre.code {
737 | margin-left: 2em;
738 | margin-right: 2em; }
739 |
740 | pre.code .ln {
741 | color: grey; }
742 |
743 | /* line numbers */
744 | pre.code, code {
745 | background-color: #eeeeee; }
746 |
747 | pre.code .comment, code .comment {
748 | color: #5c6576; }
749 |
750 | pre.code .keyword, code .keyword {
751 | color: #3B0D06;
752 | font-weight: bold; }
753 |
754 | pre.code .literal.string, code .literal.string {
755 | color: #0c5404; }
756 |
757 | pre.code .name.builtin, code .name.builtin {
758 | color: #352b84; }
759 |
760 | pre.code .deleted, code .deleted {
761 | background-color: #DEB0A1; }
762 |
763 | pre.code .inserted, code .inserted {
764 | background-color: #A3D289; }
765 |
766 | span.classifier {
767 | font-style: oblique; }
768 |
769 | span.classifier-delimiter {
770 | font-weight: bold; }
771 |
772 | span.option {
773 | white-space: nowrap; }
774 |
775 | span.problematic {
776 | color: #b30000; }
777 |
778 | span.section-subtitle {
779 | /* font-size relative to parent (h1..h6 element) */
780 | font-size: 80%; }
781 |
782 | span.DecNumber {
783 | color: var(--number); }
784 |
785 | span.BinNumber {
786 | color: var(--number); }
787 |
788 | span.HexNumber {
789 | color: var(--number); }
790 |
791 | span.OctNumber {
792 | color: var(--number); }
793 |
794 | span.FloatNumber {
795 | color: var(--number); }
796 |
797 | span.Identifier {
798 | color: var(--identifier); }
799 |
800 | span.Keyword {
801 | font-weight: 600;
802 | color: var(--keyword); }
803 |
804 | span.StringLit {
805 | color: var(--literal); }
806 |
807 | span.LongStringLit {
808 | color: var(--literal); }
809 |
810 | span.CharLit {
811 | color: var(--literal); }
812 |
813 | span.EscapeSequence {
814 | color: var(--escapeSequence); }
815 |
816 | span.Operator {
817 | color: var(--operator); }
818 |
819 | span.Punctuation {
820 | color: var(--punctuation); }
821 |
822 | span.Comment, span.LongComment {
823 | font-style: italic;
824 | font-weight: 400;
825 | color: var(--comment); }
826 |
827 | span.RegularExpression {
828 | color: darkviolet; }
829 |
830 | span.TagStart {
831 | color: darkviolet; }
832 |
833 | span.TagEnd {
834 | color: darkviolet; }
835 |
836 | span.Key {
837 | color: #252dbe; }
838 |
839 | span.Value {
840 | color: #252dbe; }
841 |
842 | span.RawData {
843 | color: var(--raw-data); }
844 |
845 | span.Assembler {
846 | color: #252dbe; }
847 |
848 | span.Preprocessor {
849 | color: #252dbe; }
850 |
851 | span.Directive {
852 | color: #252dbe; }
853 |
854 | span.Command, span.Rule, span.Hyperlink, span.Label, span.Reference,
855 | span.Other {
856 | color: var(--other); }
857 |
858 | /* Pop type, const, proc, and iterator defs in nim def blocks */
859 | dt pre > span.Identifier, dt pre > span.Operator {
860 | color: var(--identifier);
861 | font-weight: 700; }
862 |
863 | dt pre > span.Keyword ~ span.Identifier, dt pre > span.Identifier ~ span.Identifier,
864 | dt pre > span.Operator ~ span.Identifier, dt pre > span.Other ~ span.Identifier {
865 | color: var(--identifier);
866 | font-weight: inherit; }
867 |
868 | /* Nim sprite for the footer (taken from main page favicon) */
869 | .nim-sprite {
870 | display: inline-block;
871 | width: 51px;
872 | height: 14px;
873 | background-position: 0 0;
874 | background-size: 51px 14px;
875 | -webkit-filter: opacity(50%);
876 | background-repeat: no-repeat;
877 | background-image: var(--nim-sprite-base64);
878 | margin-bottom: 5px; }
879 |
880 | span.pragmadots {
881 | /* Position: relative frees us up to make the dots
882 | look really nice without fucking up the layout and
883 | causing bulging in the parent container */
884 | position: relative;
885 | /* 1px down looks slightly nicer */
886 | top: 1px;
887 | padding: 2px;
888 | background-color: var(--third-background);
889 | border-radius: 4px;
890 | margin: 0 2px;
891 | cursor: pointer;
892 | font-size: 0.8em;
893 | }
894 |
895 | span.pragmadots:hover {
896 | background-color: var(--hint);
897 | }
898 | span.pragmawrap {
899 | display: none;
900 | }
901 |
902 | span.attachedType {
903 | display: none;
904 | visibility: hidden;
905 | }
906 |
--------------------------------------------------------------------------------
/docs/stacks.html:
--------------------------------------------------------------------------------
1 |
2 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 | stacks
21 |
22 |
23 |
24 |
25 |
60 |
61 |
62 |
63 |
64 |
65 |
stacks
66 |
67 |
68 |
69 |
73 |
Dark Mode
74 |
75 |
76 |
77 | -
78 | Index
79 |
80 |
81 |
82 |
83 | Search:
85 |
86 |
87 | Group by:
88 |
92 |
93 |
94 | -
95 | Types
96 |
104 |
105 | -
106 | Procs
107 |
108 |
113 |
118 |
123 |
128 |
133 |
138 |
143 |
148 |
153 |
158 |
163 |
168 |
169 |
170 |
171 |
172 |
173 |
174 |
175 |
176 |
177 |
178 |
Pure Nim stack implementation based on sequences.
179 |
Example:
180 |
181 |
182 | import stacks
183 |
184 | let a = "Hello, World!"
185 | var s = Stack[char]()
186 | for letter in a:
187 | s.push(letter)
188 |
189 | var b: string
190 | while not s.isEmpty:
191 | b.add(s.pop)
192 |
193 | assert b == "!dlroW ,olleH"
194 |
195 |
196 |
197 |
198 | EStackEmpty = object of CatchableError
199 | -
200 |
201 |
202 |
203 |
204 |
205 | Stack[T] = object
206 | data: seq[T]
207 |
208 | -
209 |
210 |
211 |
212 |
213 |
214 |
215 |
216 |
217 |
218 |
219 | proc newStack[T](capacity = 8): Stack[T]
220 | -
221 |
222 | Creates a new stack. Optionally, the initial capacity can be reserved via capacity parameter.
var a = newStack[int](capacity = 64)
223 |
224 |
225 |
226 | proc len[T](s: Stack[T]): int
227 | -
228 |
229 | Returns the number of elements in the stack. Returns 0 if the stack is empty.
var a = newStack[int]()
230 | assert a.len == 0
231 | a.push(10); a.push(20)
232 | assert a.len == 2
233 |
234 |
235 |
236 | proc empty[T](s: Stack[T]): bool {...}{.deprecated: "use isEmpty() instead".}
237 | -
238 |
239 | Deprecated: use isEmpty() instead
240 |
241 |
242 |
243 |
244 |
245 |
246 | proc isEmpty[T](s: Stack[T]): bool
247 | -
248 |
249 | Returns true if stack contains no elements, false otherwise.
var a = newStack[int]()
250 | assert a.isEmpty == true
251 | a.push(10)
252 | assert a.isEmpty == false
253 |
254 |
255 |
256 | proc push[T](s: var Stack[T]; element: T)
257 | -
258 |
259 | Pushes element onto the top of the stack.
var a = newStack[int]()
260 | a.push(10)
261 |
262 |
263 |
264 | proc pop[T](s: var Stack[T]): T {...}{.raises: [EStackEmpty].}
265 | -
266 |
267 | Pops the top element from the stack. Raises EStackEmpty exception if the stack is empty.
var a = newStack[int]()
268 | a.push(10)
269 | discard a.pop()
270 | doAssertRaises(EStackEmpty, echo a.pop())
271 |
272 |
273 |
274 | proc popUnsafe[T](s: var Stack[T]): T
275 | -
276 |
277 | Pops the top element from the stack without checking if it's not empty. Make sure the stack is not empty.
var a = newStack[int]()
278 | a.push(10)
279 | check(a.popUnsafe() == 10)
280 |
281 |
282 |
283 | proc peek[T](s: Stack[T]): T {...}{.raises: [EStackEmpty].}
284 | -
285 |
286 | Peeks the top element from the stack. Raises EStackEmpty exception if the stack is empty.
var a = newStack[int]()
287 | a.push(10)
288 | check(a.peek() == 10)
289 |
290 |
291 |
292 | proc peekUnsafe[T](s: Stack[T]): T
293 | -
294 |
295 | Peeks the top element from the stack without checking if it's not empty. Make sure the stack is not empty.
var a = newStack[int]()
296 | a.push(10)
297 | check(a.peekUnsafe() == 10)
298 |
299 |
300 |
301 | proc clear[T](s: var Stack[T])
302 | -
303 |
304 | Empties the stack. Does nothing if the stack is already empty.
var a = newStack[int]()
305 | a.push(10)
306 | a.clear()
307 | assert a.isEmpty == true
308 |
309 |
310 |
311 | proc toSeq[T](s: Stack[T]): seq[T]
312 | -
313 |
314 | Returns sequence representation of a stack.
var a = newStack[int]()
315 | a.push(10); a.push(20)
316 | assert a.toSeq() == @[10, 20]
317 |
318 |
319 |
320 | proc `$`[T](s: Stack[T]): string
321 | -
322 |
323 | Returns string representation of a stack
var a = newStack[int]()
324 | a.push(10); a.push(20)
325 | assert $a == "Stack[10, 20]"
326 |
327 |
328 |
329 |
330 |
331 |
332 |
333 |
334 |
335 |
336 |
337 |
338 | Made with Nim. Generated: 2020-10-09 18:03:35 UTC
339 |
340 |
341 |
342 |
343 |
344 |
345 |
346 |
--------------------------------------------------------------------------------
/src/stacks.nim:
--------------------------------------------------------------------------------
1 | ## Pure Nim stack implementation based on sequences.
2 | ##
3 | ## **Example:**
4 | ##
5 | ## .. code-block:: Nim
6 | ## # Reverting a string using a stack
7 | ##
8 | ## import stacks
9 | ##
10 | ## let a = "Hello, World!"
11 | ## var s = Stack[char]()
12 | ## for letter in a:
13 | ## s.push(letter)
14 | ##
15 | ## var b: string
16 | ## while not s.isEmpty:
17 | ## b.add(s.pop)
18 | ##
19 | ## assert b == "!dlroW ,olleH"
20 | ##
21 |
22 | import math
23 |
24 | type EStackEmpty* = object of CatchableError
25 |
26 | type
27 | Stack* [T] = object
28 | data: seq[T]
29 |
30 | proc newStack* [T](capacity = 8): Stack[T] =
31 | ## Creates a new stack.
32 | ## Optionally, the initial capacity can be reserved via `capacity` parameter.
33 | ##
34 | ## .. code-block:: Nim
35 | ## var a = newStack[int](capacity = 64)
36 | assert isPowerOfTwo(capacity)
37 | result.data = newSeqOfCap[T](capacity)
38 |
39 | proc len* [T](s: Stack[T]): int =
40 | ## Returns the number of elements in the stack.
41 | ## Returns `0` if the stack is empty.
42 | ##
43 | ## .. code-block:: Nim
44 | ## var a = newStack[int]()
45 | ## assert a.len == 0
46 | ## a.push(10); a.push(20)
47 | ## assert a.len == 2
48 | s.data.len()
49 |
50 | proc empty* [T](s: Stack[T]): bool {.deprecated: "use isEmpty() instead".} =
51 | s.data.len() == 0
52 |
53 | proc isEmpty* [T](s: Stack[T]): bool =
54 | ## Returns `true` if stack contains no elements, `false` otherwise.
55 | ##
56 | ## .. code-block:: Nim
57 | ## var a = newStack[int]()
58 | ## assert a.isEmpty == true
59 | ## a.push(10)
60 | ## assert a.isEmpty == false
61 | s.data.len() == 0
62 |
63 | proc push* [T](s: var Stack[T], element: T) =
64 | ## Pushes `element` onto the top of the stack.
65 | ##
66 | ## .. code-block:: Nim
67 | ## var a = newStack[int]()
68 | ## a.push(10)
69 | s.data.add(element)
70 |
71 | proc pop* [T](s: var Stack[T]): T {.raises: [EStackEmpty].} =
72 | ## Pops the top element from the stack.
73 | ## Raises `EStackEmpty` exception if the stack is empty.
74 | ##
75 | ## .. code-block:: Nim
76 | ## var a = newStack[int]()
77 | ## a.push(10)
78 | ## discard a.pop()
79 | ## doAssertRaises(EStackEmpty, echo a.pop())
80 | if not s.isEmpty:
81 | result = s.data[^1]
82 | s.data.setLen s.data.len - 1
83 | else:
84 | raise newException(EStackEmpty, "Cannot pop an empty stack")
85 |
86 | proc popUnsafe* [T](s: var Stack[T]): T =
87 | ## Pops the top element from the stack without checking if it's not empty.
88 | ## Make sure the stack is not empty.
89 | ##
90 | ## .. code-block:: Nim
91 | ## var a = newStack[int]()
92 | ## a.push(10)
93 | ## check(a.popUnsafe() == 10)
94 | result = s.data[^1]
95 | s.data.setLen s.data.len - 1
96 |
97 | proc peek* [T](s: Stack[T]): T {.raises: [EStackEmpty].} =
98 | ## Peeks the top element from the stack.
99 | ## Raises `EStackEmpty` exception if the stack is empty.
100 | ##
101 | ## .. code-block:: Nim
102 | ## var a = newStack[int]()
103 | ## a.push(10)
104 | ## check(a.peek() == 10)
105 | if not s.isEmpty:
106 | result = s.data[^1]
107 | else:
108 | raise newException(EStackEmpty, "Cannot peek an empty stack")
109 |
110 | proc peekUnsafe* [T](s: Stack[T]): T =
111 | ## Peeks the top element from the stack without checking if it's not empty.
112 | ## Make sure the stack is not empty.
113 | ##
114 | ## .. code-block:: Nim
115 | ## var a = newStack[int]()
116 | ## a.push(10)
117 | ## check(a.peekUnsafe() == 10)
118 | s.data[^1]
119 |
120 | proc clear* [T](s: var Stack[T]) =
121 | ## Empties the stack. Does nothing if the stack is already empty.
122 | ##
123 | ## .. code-block:: Nim
124 | ## var a = newStack[int]()
125 | ## a.push(10)
126 | ## a.clear()
127 | ## assert a.isEmpty == true
128 | if not s.isEmpty:
129 | s.data.setLen 0
130 |
131 | proc toSeq* [T](s: Stack[T]): seq[T] =
132 | ## Returns sequence representation of a stack.
133 | ##
134 | ## .. code-block:: Nim
135 | ## var a = newStack[int]()
136 | ## a.push(10); a.push(20)
137 | ## assert a.toSeq() == @[10, 20]
138 | s.data
139 |
140 | proc `$`* [T](s: Stack[T]): string =
141 | ## Returns string representation of a stack
142 | ##
143 | ## .. code-block:: Nim
144 | ## var a = newStack[int]()
145 | ## a.push(10); a.push(20)
146 | ## assert $a == "Stack[10, 20]"
147 | result = "Stack["
148 | if not s.isEmpty():
149 | for i in 0 .. s.data.high() - 1:
150 | result &= $s.data[i]
151 | result &= ", "
152 | result &= $s.data[^1]
153 | result &= "]"
154 |
--------------------------------------------------------------------------------
/stacks.nimble:
--------------------------------------------------------------------------------
1 | # Package
2 |
3 | version = "0.4.2"
4 | author = "Max Skybin"
5 | description = "Pure Nim stack implementation based on sequences."
6 | license = "MIT"
7 | srcDir = "src"
8 |
9 | # Dependencies
10 |
11 | requires "nim >= 1.2.4"
12 |
--------------------------------------------------------------------------------
/tests/config.nims:
--------------------------------------------------------------------------------
1 | switch("path", "$projectDir/../src")
--------------------------------------------------------------------------------
/tests/test_stacks.nim:
--------------------------------------------------------------------------------
1 | # To run these tests, execute `nimble test`.
2 | import unittest
3 | import stacks
4 |
5 | test "can create an empty stack":
6 | var a = newStack[int]()
7 | check a.len() == 0
8 |
9 | test "can create an empty stack with capacity":
10 | var a = newStack[int](capacity = 32)
11 | check a.len() == 0
12 |
13 | test "can push into stack":
14 | var a = newStack[int]()
15 | a.push(10)
16 | check a.len == 1
17 |
18 | test "can pop from non-empty stack":
19 | var a = newStack[int]()
20 | a.push(10)
21 | check a.pop() == 10
22 | a.push(11)
23 | check a.popUnsafe() == 11
24 |
25 | test "can popUnsafe from non-empty stack":
26 | var a = newStack[int]()
27 | a.push(10)
28 | check a.popUnsafe() == 10
29 |
30 | test "can peek from non-empty stack":
31 | var a = newStack[int]()
32 | a.push(10)
33 | check a.peek() == 10
34 |
35 | test "can peekUnsafe from non-empty stack":
36 | var a = newStack[int]()
37 | a.push(10)
38 | check a.peekUnsafe() == 10
39 |
40 | test "pop on empty stack raises exception":
41 | var a = newStack[int]()
42 | expect EStackEmpty:
43 | discard a.pop()
44 |
45 | test "can clear stack":
46 | var a = newStack[int]()
47 | a.push(10)
48 | a.clear()
49 | check a.len() == 0
50 | check a.isEmpty() == true
51 |
52 | test "can correctly check if stack is empty":
53 | var a = newStack[int]()
54 | a.push(10)
55 | check a.isEmpty() == false
56 | a.clear()
57 | check a.isEmpty() == true
58 |
59 | test "can get sequence representation of the stack":
60 | var a = newStack[int]()
61 | a.push(10)
62 | a.push(20)
63 | a.push(30)
64 | check a.toSeq() == @[10, 20, 30]
65 |
66 | test "can get string representation of the stack":
67 | var a = newStack[int]()
68 | a.push(10)
69 | a.push(20)
70 | a.push(30)
71 | check $a == "Stack[10, 20, 30]"
72 |
--------------------------------------------------------------------------------