├── .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 | 82 |
83 | Search: 85 |
86 |
87 | Group by: 88 | 92 |
93 | 173 | 174 |
175 |
176 |
177 | 178 |

Pure Nim stack implementation based on sequences.

179 |

Example:

180 |
# Reverting a string using a stack
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 |

Types

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 |

Procs

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 | 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 | --------------------------------------------------------------------------------