diff --git a/.github/images/logo/PicoPortal.png b/.github/images/logo/PicoPortal.png new file mode 100644 index 0000000..56696cf Binary files /dev/null and b/.github/images/logo/PicoPortal.png differ diff --git a/.github/images/logo/PicoPortal.psd b/.github/images/logo/PicoPortal.psd new file mode 100644 index 0000000..6335347 Binary files /dev/null and b/.github/images/logo/PicoPortal.psd differ diff --git a/.github/images/ng-icons/Angular Material Icons.url b/.github/images/ng-icons/Angular Material Icons.url new file mode 100644 index 0000000..9edb26d --- /dev/null +++ b/.github/images/ng-icons/Angular Material Icons.url @@ -0,0 +1,5 @@ +[{000214A0-0000-0000-C000-000000000046}] +Prop3=19,11 +[InternetShortcut] +IDList= +URL=https://fonts.google.com/icons diff --git a/.github/images/ng-icons/email.svg b/.github/images/ng-icons/email.svg new file mode 100644 index 0000000..12b910c --- /dev/null +++ b/.github/images/ng-icons/email.svg @@ -0,0 +1,12 @@ + + Email + + diff --git a/.github/images/ng-icons/info.svg b/.github/images/ng-icons/info.svg new file mode 100644 index 0000000..47d2179 --- /dev/null +++ b/.github/images/ng-icons/info.svg @@ -0,0 +1,15 @@ + diff --git a/.github/images/ng-icons/warn.svg b/.github/images/ng-icons/warn.svg new file mode 100644 index 0000000..c514e8a --- /dev/null +++ b/.github/images/ng-icons/warn.svg @@ -0,0 +1,15 @@ + diff --git a/.github/images/previews/mini_red.png b/.github/images/previews/mini_red.png new file mode 100644 index 0000000..d89ea83 Binary files /dev/null and b/.github/images/previews/mini_red.png differ diff --git a/.github/images/previews/xl_blue.png b/.github/images/previews/xl_blue.png new file mode 100644 index 0000000..7bfdad1 Binary files /dev/null and b/.github/images/previews/xl_blue.png differ diff --git a/.github/images/simple-icons/Simple Icons.url b/.github/images/simple-icons/Simple Icons.url new file mode 100644 index 0000000..2da5557 --- /dev/null +++ b/.github/images/simple-icons/Simple Icons.url @@ -0,0 +1,2 @@ +[InternetShortcut] +URL=https://github.com/simple-icons diff --git a/.github/images/simple-icons/bitcoin-btc-logo.svg b/.github/images/simple-icons/bitcoin-btc-logo.svg new file mode 100644 index 0000000..2b75c99 --- /dev/null +++ b/.github/images/simple-icons/bitcoin-btc-logo.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + diff --git a/.github/images/simple-icons/buymeacoffee.svg b/.github/images/simple-icons/buymeacoffee.svg new file mode 100644 index 0000000..608ec8f --- /dev/null +++ b/.github/images/simple-icons/buymeacoffee.svg @@ -0,0 +1 @@ +Buy Me A Coffee \ No newline at end of file diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml new file mode 100644 index 0000000..c9138ba --- /dev/null +++ b/.github/workflows/lint.yml @@ -0,0 +1,23 @@ +name: Python Linting + +on: + push: + pull_request: + schedule: + # Runs every 7 days at midnight UTC + - cron: '0 0 */7 * *' + +jobs: + build: + name: Python Linting + runs-on: ubuntu-20.04 + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Install Python Dependencies + run: npm run lint:install + + - name: Lint Python Code + run: npm run lint diff --git a/.gitignore b/.gitignore index 82f9275..623adae 100644 --- a/.gitignore +++ b/.gitignore @@ -160,3 +160,73 @@ cython_debug/ # and can be added to the global gitignore or merged into this file. For a more nuclear # option (not recommended) you can uncomment the following to ignore the entire idea folder. #.idea/ + +# Root logs +/*.log + +# Compiled output +/tmp +/out-tsc +/bazel-out + +# Node +/node_modules +npm-debug.log +yarn-error.log + +# IDEs and editors +.idea/ +.project +.classpath +.c9/ +*.launch +.settings/ +*.sublime-workspace + +# IDE - Visual Studio +.vs/* + +# Visual Studio Code +.vscode/* +!.vscode/settings.json +!.vscode/tasks.json +!.vscode/launch.json +!.vscode/extensions.json +.history/* + +# Husky Local Environment Files +/.husky/_ +/.husky/_/**/* + +# Miscellaneous +/.angular/cache +.sass-cache/ +/connect.lock +/coverage +/libpeerconnection.log +testem.log +/typings + +# System files +.DS_Store +Thumbs.db + +# Build and Release Folders +bin-debug/ +bin-release/ +[Oo]bj/ +[Bb]in/ + +# Other files and folders +.settings/ + +# Executables +*.swf +*.air +*.ipa +*.apk +*.csr +*.pem + +# src modules +src/modules/ diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml new file mode 100644 index 0000000..3d6e40c --- /dev/null +++ b/.pre-commit-config.yaml @@ -0,0 +1,15 @@ +repos: + # black + - repo: https://github.com/psf/black + rev: 24.8.0 + hooks: + - id: black + args: ['src/'] + exclude: 'src/modules/.*' + # flake8 + - repo: https://github.com/PyCQA/flake8 + rev: 7.1.1 + hooks: + - id: flake8 + args: ['--show-source', '--ignore', 'E501'] + exclude: 'src/modules/.*' diff --git a/LICENSE.md b/LICENSE.md new file mode 100644 index 0000000..e4cf43e --- /dev/null +++ b/LICENSE.md @@ -0,0 +1,159 @@ +# Attribution-NonCommercial 4.0 International + +> *Creative Commons Corporation (“Creative Commons”) is not a law firm and does not provide legal services or legal advice. Distribution of Creative Commons public licenses does not create a lawyer-client or other relationship. Creative Commons makes its licenses and related information available on an “as-is” basis. Creative Commons gives no warranties regarding its licenses, any material licensed under their terms and conditions, or any related information. Creative Commons disclaims all liability for damages resulting from their use to the fullest extent possible.* +> +> ### Using Creative Commons Public Licenses +> +> Creative Commons public licenses provide a standard set of terms and conditions that creators and other rights holders may use to share original works of authorship and other material subject to copyright and certain other rights specified in the public license below. The following considerations are for informational purposes only, are not exhaustive, and do not form part of our licenses. +> +> * __Considerations for licensors:__ Our public licenses are intended for use by those authorized to give the public permission to use material in ways otherwise restricted by copyright and certain other rights. Our licenses are irrevocable. Licensors should read and understand the terms and conditions of the license they choose before applying it. Licensors should also secure all rights necessary before applying our licenses so that the public can reuse the material as expected. Licensors should clearly mark any material not subject to the license. This includes other CC-licensed material, or material used under an exception or limitation to copyright. [More considerations for licensors](http://wiki.creativecommons.org/Considerations_for_licensors_and_licensees#Considerations_for_licensors). +> +> * __Considerations for the public:__ By using one of our public licenses, a licensor grants the public permission to use the licensed material under specified terms and conditions. If the licensor’s permission is not necessary for any reason–for example, because of any applicable exception or limitation to copyright–then that use is not regulated by the license. Our licenses grant only permissions under copyright and certain other rights that a licensor has authority to grant. Use of the licensed material may still be restricted for other reasons, including because others have copyright or other rights in the material. A licensor may make special requests, such as asking that all changes be marked or described. Although not required by our licenses, you are encouraged to respect those requests where reasonable. [More considerations for the public](http://wiki.creativecommons.org/Considerations_for_licensors_and_licensees#Considerations_for_licensees). + +## Creative Commons Attribution-NonCommercial 4.0 International Public License + +By exercising the Licensed Rights (defined below), You accept and agree to be bound by the terms and conditions of this Creative Commons Attribution-NonCommercial 4.0 International Public License ("Public License"). To the extent this Public License may be interpreted as a contract, You are granted the Licensed Rights in consideration of Your acceptance of these terms and conditions, and the Licensor grants You such rights in consideration of benefits the Licensor receives from making the Licensed Material available under these terms and conditions. + +### Section 1 – Definitions. + +a. __Adapted Material__ means material subject to Copyright and Similar Rights that is derived from or based upon the Licensed Material and in which the Licensed Material is translated, altered, arranged, transformed, or otherwise modified in a manner requiring permission under the Copyright and Similar Rights held by the Licensor. For purposes of this Public License, where the Licensed Material is a musical work, performance, or sound recording, Adapted Material is always produced where the Licensed Material is synched in timed relation with a moving image. + +b. __Adapter's License__ means the license You apply to Your Copyright and Similar Rights in Your contributions to Adapted Material in accordance with the terms and conditions of this Public License. + +c. __Copyright and Similar Rights__ means copyright and/or similar rights closely related to copyright including, without limitation, performance, broadcast, sound recording, and Sui Generis Database Rights, without regard to how the rights are labeled or categorized. For purposes of this Public License, the rights specified in Section 2(b)(1)-(2) are not Copyright and Similar Rights. + +d. __Effective Technological Measures__ means those measures that, in the absence of proper authority, may not be circumvented under laws fulfilling obligations under Article 11 of the WIPO Copyright Treaty adopted on December 20, 1996, and/or similar international agreements. + +e. __Exceptions and Limitations__ means fair use, fair dealing, and/or any other exception or limitation to Copyright and Similar Rights that applies to Your use of the Licensed Material. + +f. __Licensed Material__ means the artistic or literary work, database, or other material to which the Licensor applied this Public License. + +g. __Licensed Rights__ means the rights granted to You subject to the terms and conditions of this Public License, which are limited to all Copyright and Similar Rights that apply to Your use of the Licensed Material and that the Licensor has authority to license. + +h. __Licensor__ means the individual(s) or entity(ies) granting rights under this Public License. + +i. __NonCommercial__ means not primarily intended for or directed towards commercial advantage or monetary compensation. For purposes of this Public License, the exchange of the Licensed Material for other material subject to Copyright and Similar Rights by digital file-sharing or similar means is NonCommercial provided there is no payment of monetary compensation in connection with the exchange. + +j. __Share__ means to provide material to the public by any means or process that requires permission under the Licensed Rights, such as reproduction, public display, public performance, distribution, dissemination, communication, or importation, and to make material available to the public including in ways that members of the public may access the material from a place and at a time individually chosen by them. + +k. __Sui Generis Database Rights__ means rights other than copyright resulting from Directive 96/9/EC of the European Parliament and of the Council of 11 March 1996 on the legal protection of databases, as amended and/or succeeded, as well as other essentially equivalent rights anywhere in the world. + +l. __You__ means the individual or entity exercising the Licensed Rights under this Public License. Your has a corresponding meaning. + +### Section 2 – Scope. + +a. ___License grant.___ + + 1. Subject to the terms and conditions of this Public License, the Licensor hereby grants You a worldwide, royalty-free, non-sublicensable, non-exclusive, irrevocable license to exercise the Licensed Rights in the Licensed Material to: + + A. reproduce and Share the Licensed Material, in whole or in part, for NonCommercial purposes only; and + + B. produce, reproduce, and Share Adapted Material for NonCommercial purposes only. + + 2. __Exceptions and Limitations.__ For the avoidance of doubt, where Exceptions and Limitations apply to Your use, this Public License does not apply, and You do not need to comply with its terms and conditions. + + 3. __Term.__ The term of this Public License is specified in Section 6(a). + + 4. __Media and formats; technical modifications allowed.__ The Licensor authorizes You to exercise the Licensed Rights in all media and formats whether now known or hereafter created, and to make technical modifications necessary to do so. The Licensor waives and/or agrees not to assert any right or authority to forbid You from making technical modifications necessary to exercise the Licensed Rights, including technical modifications necessary to circumvent Effective Technological Measures. For purposes of this Public License, simply making modifications authorized by this Section 2(a)(4) never produces Adapted Material. + + 5. __Downstream recipients.__ + + A. __Offer from the Licensor – Licensed Material.__ Every recipient of the Licensed Material automatically receives an offer from the Licensor to exercise the Licensed Rights under the terms and conditions of this Public License. + + B. __No downstream restrictions.__ You may not offer or impose any additional or different terms or conditions on, or apply any Effective Technological Measures to, the Licensed Material if doing so restricts exercise of the Licensed Rights by any recipient of the Licensed Material. + + 6. __No endorsement.__ Nothing in this Public License constitutes or may be construed as permission to assert or imply that You are, or that Your use of the Licensed Material is, connected with, or sponsored, endorsed, or granted official status by, the Licensor or others designated to receive attribution as provided in Section 3(a)(1)(A)(i). + +b. ___Other rights.___ + + 1. Moral rights, such as the right of integrity, are not licensed under this Public License, nor are publicity, privacy, and/or other similar personality rights; however, to the extent possible, the Licensor waives and/or agrees not to assert any such rights held by the Licensor to the limited extent necessary to allow You to exercise the Licensed Rights, but not otherwise. + + 2. Patent and trademark rights are not licensed under this Public License. + + 3. To the extent possible, the Licensor waives any right to collect royalties from You for the exercise of the Licensed Rights, whether directly or through a collecting society under any voluntary or waivable statutory or compulsory licensing scheme. In all other cases the Licensor expressly reserves any right to collect such royalties, including when the Licensed Material is used other than for NonCommercial purposes. + +### Section 3 – License Conditions. + +Your exercise of the Licensed Rights is expressly made subject to the following conditions. + +a. ___Attribution.___ + + 1. If You Share the Licensed Material (including in modified form), You must: + + A. retain the following if it is supplied by the Licensor with the Licensed Material: + + i. identification of the creator(s) of the Licensed Material and any others designated to receive attribution, in any reasonable manner requested by the Licensor (including by pseudonym if designated); + + ii. a copyright notice; + + iii. a notice that refers to this Public License; + + iv. a notice that refers to the disclaimer of warranties; + + v. a URI or hyperlink to the Licensed Material to the extent reasonably practicable; + + B. indicate if You modified the Licensed Material and retain an indication of any previous modifications; and + + C. indicate the Licensed Material is licensed under this Public License, and include the text of, or the URI or hyperlink to, this Public License. + + 2. You may satisfy the conditions in Section 3(a)(1) in any reasonable manner based on the medium, means, and context in which You Share the Licensed Material. For example, it may be reasonable to satisfy the conditions by providing a URI or hyperlink to a resource that includes the required information. + + 3. If requested by the Licensor, You must remove any of the information required by Section 3(a)(1)(A) to the extent reasonably practicable. + + 4. If You Share Adapted Material You produce, the Adapter's License You apply must not prevent recipients of the Adapted Material from complying with this Public License. + +### Section 4 – Sui Generis Database Rights. + +Where the Licensed Rights include Sui Generis Database Rights that apply to Your use of the Licensed Material: + +a. for the avoidance of doubt, Section 2(a)(1) grants You the right to extract, reuse, reproduce, and Share all or a substantial portion of the contents of the database for NonCommercial purposes only; + +b. if You include all or a substantial portion of the database contents in a database in which You have Sui Generis Database Rights, then the database in which You have Sui Generis Database Rights (but not its individual contents) is Adapted Material; and + +c. You must comply with the conditions in Section 3(a) if You Share all or a substantial portion of the contents of the database. + +For the avoidance of doubt, this Section 4 supplements and does not replace Your obligations under this Public License where the Licensed Rights include other Copyright and Similar Rights. + +### Section 5 – Disclaimer of Warranties and Limitation of Liability. + +a. __Unless otherwise separately undertaken by the Licensor, to the extent possible, the Licensor offers the Licensed Material as-is and as-available, and makes no representations or warranties of any kind concerning the Licensed Material, whether express, implied, statutory, or other. This includes, without limitation, warranties of title, merchantability, fitness for a particular purpose, non-infringement, absence of latent or other defects, accuracy, or the presence or absence of errors, whether or not known or discoverable. Where disclaimers of warranties are not allowed in full or in part, this disclaimer may not apply to You.__ + +b. __To the extent possible, in no event will the Licensor be liable to You on any legal theory (including, without limitation, negligence) or otherwise for any direct, special, indirect, incidental, consequential, punitive, exemplary, or other losses, costs, expenses, or damages arising out of this Public License or use of the Licensed Material, even if the Licensor has been advised of the possibility of such losses, costs, expenses, or damages. Where a limitation of liability is not allowed in full or in part, this limitation may not apply to You.__ + +c. The disclaimer of warranties and limitation of liability provided above shall be interpreted in a manner that, to the extent possible, most closely approximates an absolute disclaimer and waiver of all liability. + +### Section 6 – Term and Termination. + +a. This Public License applies for the term of the Copyright and Similar Rights licensed here. However, if You fail to comply with this Public License, then Your rights under this Public License terminate automatically. + +b. Where Your right to use the Licensed Material has terminated under Section 6(a), it reinstates: + + 1. automatically as of the date the violation is cured, provided it is cured within 30 days of Your discovery of the violation; or + + 2. upon express reinstatement by the Licensor. + + For the avoidance of doubt, this Section 6(b) does not affect any right the Licensor may have to seek remedies for Your violations of this Public License. + +c. For the avoidance of doubt, the Licensor may also offer the Licensed Material under separate terms or conditions or stop distributing the Licensed Material at any time; however, doing so will not terminate this Public License. + +d. Sections 1, 5, 6, 7, and 8 survive termination of this Public License. + +### Section 7 – Other Terms and Conditions. + +a. The Licensor shall not be bound by any additional or different terms or conditions communicated by You unless expressly agreed. + +b. Any arrangements, understandings, or agreements regarding the Licensed Material not stated herein are separate from and independent of the terms and conditions of this Public License. + +### Section 8 – Interpretation. + +a. For the avoidance of doubt, this Public License does not, and shall not be interpreted to, reduce, limit, restrict, or impose conditions on any use of the Licensed Material that could lawfully be made without permission under this Public License. + +b. To the extent possible, if any provision of this Public License is deemed unenforceable, it shall be automatically reformed to the minimum extent necessary to make it enforceable. If the provision cannot be reformed, it shall be severed from this Public License without affecting the enforceability of the remaining terms and conditions. + +c. No term or condition of this Public License will be waived and no failure to comply consented to unless expressly agreed to by the Licensor. + +d. Nothing in this Public License constitutes or may be interpreted as a limitation upon, or waiver of, any privileges and immunities that apply to the Licensor or You, including from the legal processes of any jurisdiction or authority. + +> Creative Commons is not a party to its public licenses. Notwithstanding, Creative Commons may elect to apply one of its public licenses to material it publishes and in those instances will be considered the “Licensor.” Except for the limited purpose of indicating that material is shared under a Creative Commons public license or as otherwise permitted by the Creative Commons policies published at [creativecommons.org/policies](http://creativecommons.org/policies), Creative Commons does not authorize the use of the trademark “Creative Commons” or any other trademark or logo of Creative Commons without its prior written consent including, without limitation, in connection with any unauthorized modifications to any of its public licenses or any other arrangements, understandings, or agreements concerning use of licensed material. For the avoidance of doubt, this paragraph does not form part of the public licenses. +> +> Creative Commons may be contacted at creativecommons.org \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..83eddcd --- /dev/null +++ b/README.md @@ -0,0 +1,338 @@ +
+ +

Pico Portal

+

+ Turn your Raspberry Pi Pico W into a portable, powerful Wi-Fi access point with this lightweight captive portal software. + Serve web content/pages and display real-time connection info directly on the onboard Pimoroni screen. Log all the connection details for later debugging and use! +

+

+ Whether you're testing networks, showcasing web projects, or exploring IoT, this tool gives you the flexibility to do it all. + It’s easily adaptable for various purposes—serve web applications, demo single-page apps (SPAs), or set up captive portals for network security testing. +

+
+ +## Index + +- [Preview images](#previews) +- [Hardware](#hardware) + - [Purchase](#purchase-device) + - [Build your own](#build-your-own) +- [Firmware Setup](#firmware-setup) + - [Connecting to PC](#connecting) + - [Installing Firmware](#installing-firmware) +- [Software Setup](#software-setup) + - [Installing Software](#installing-software) + - [User Defined Settings](#user-defined-settings) + - [Button Functions](#button-functions) +- [Development](#development) + - [Requirements](#requirements) + - [Development Setup](#development-setup) + - [Scripts](#scripts) +- [Licensing](#licensing) +- [Wrapping Up](#wrapping-up) + + + + + +## Preview images + +### Pico Portal XL + +![Pico Portal XL][img-pico-portal-xl-blue] + +### Pico Portal Mini + +![Pico Portal Mini][img-pico-portal-mini-red] + +

[ Index ]

+ + + + + +## Hardware + +### Purchase device + +You can purchase a fully assembled Pico Portal XL, Pico Portal Mini, or a 3D-printed enclosure/case for your own build from Lambda Guru: + +🔗 https://www.lambda.guru/ + +> ![Info][img-info] **Note:** The 3D print files are available on [Lambda Guru][url-lambda-guru] for a small fee (the first time I've ever charged for models, see my free collection [here][url-free-3d]). The enclosures/cases on Lambda Guru are specifically designed for the parts listed below. + +> ![Info][img-info] **Note:** If you'd like to do your own custom build there are other alternative 3D printable cases out there on websites such as [thingiverse][url-thingiverse]. Be sure to show off a picture of your build by submitting a ticket [here][url-new-issue]! Your images will likely be shared on places like the Lambda Guru website or this repository for the community to see! + +### Build your own + +To replicate the original Pico Portal devices found on [Lambda Guru][url-lambda-guru], you will need the following: + +**Pico Portal XL** parts list: + +| Part | Link | +| :--- | :--- | +| Raspberry Pi Pico W | [ThePiHut](https://thepihut.com/products/raspberry-pi-pico-w?variant=41952994787523) | +| Pimoroni Pico Display Pack 2.0 | [Pimoroni](https://shop.pimoroni.com/products/pico-display-pack-2-0) | +| 3D Printed Enclosure (XL) | [Lambda Guru](https://www.lambda.guru/) | +| 1600 mAh LiPo Battery | — | +| LiPo SHIM | [Pimoroni](https://shop.pimoroni.com/products/pico-lipo-shim?variant=32369543086163) | + +**Pico Portal Mini** parts list: + +| Part | Link | +| :--- | :--- | +| Raspberry Pi Pico W | [ThePiHut](https://thepihut.com/products/raspberry-pi-pico-w?variant=41952994787523) | +| Pimoroni Pico Display Pack | [Pimoroni](https://shop.pimoroni.com/products/pico-display-pack) | +| 3D Printed Enclosure (Mini) | [Lambda Guru](https://www.lambda.guru/) | +| 600 mAh LiPo Battery | — | +| Pico-UPS-B | [Waveshare](https://www.waveshare.com/pico-ups-b.htm) | + +> ![Info][img-info] **Note:** This project can be ran standalone on a Raspberry Pi Pico W! Additional hardware is only required for the portable version. + +

[ Index ]

+ + + + + +## Firmware Setup + +### Connecting to PC + +To connect your Raspberry Pi Pico W to your PC for firmware installation, follow these steps: + +1. Make sure your Raspberry Pi Pico is not connected to any power source. + +2. Hold down the BOOTSEL button on your Pico. + +3. Connect the Pico to your computer using a Micro USB cable. + +4. Release the BOOTSEL button. The device should appear on your computer as a USB Mass Storage Device. + +> ![Info][img-info] **Note:** You only need to do this for Firmware installation. After the firmware is installed, you can connect your Pico to your computer without holding the BOOTSEL button. + +### Installing the Pimoroni MicroPython Firmware + +To install MicroPython on your Raspberry Pi Pico W after [connecting to your computer](#connecting-to-computer), follow these steps: + +1. Download the latest Pimoroni Pico W UF2 file "picow-vXX.YY.ZZ-pimoroni-micropython.uf2" from the official releases: + + - https://github.com/pimoroni/pimoroni-pico/releases + +2. Connect your Raspberry Pi Pico W to your PC, see [Connecting to PC](#connecting-to-pc). + +3. Drag and drop the UF2 file onto the RPI-RP2 drive. This will program the MicroPython firmware onto your Pico. + +4. Wait for a few seconds. The board will automatically reboot. Your Pico will now be running MicroPython. + +

[ Index ]

+ + + + + +## Software Setup + +### Installing Software + +> ![Info][img-info] **Note:** This project uses Node.js. Make sure you have Node.js installed on your system before proceeding. + +1. Install project dependencies: + + ```bash + npm install + ``` + +2. Install Thonny IDE: + + - https://thonny.org/ + +3. Open Thonny IDE and connect to your Raspberry Pi Pico W via USB. + +4. Copy the contents from `src/` (this repo) to the root of your Raspberry Pi Pico W using the Thonny IDE. + + > ![Info][img-info] **Note:** Be sure the "src/modules/" is copied and that the folder exists. + +5. Unplug your Raspberry Pi Pico W from your computer and connect it to a power source. + +6. Your Raspberry Pi Pico W will now boot up and display the Pico Portal interface on the Pimoroni screen. + +### User Defined Settings + +You can customize the Pico Portal settings by editing the `src/options.py` file. The settings are as follows: + +```python +{ + "wifi_ssid": "WiFi", + "wifi_password": "", + "wifi_domain": "setup.local", + "display_type": "DISPLAY_PICO_DISPLAY" + "enable_timestamps": False, + "led_brightness": 0.25 +} +``` + +| Setting | Description | +| :------ | :---------- | +| `wifi_ssid` | The SSID of the Wi-Fi network you want to create. | +| `wifi_password` | The password for the Wi-Fi network you want to create. Make sure your password is 8+ characters. Also cycle the power on and off if you change the password to fully update it. Leave blank for an open network (default). | +| `wifi_domain` | The domain name for the captive portal displayed on the connecting device. | +| `display_type` | The type of display you are using. Options are `DISPLAY_PICO_DISPLAY` (default) or `DISPLAY_PICO_DISPLAY_2`. If you don't have a screen, you can use either. | +| `enable_timestamps` | Enable or disable timestamps for the log. | +| `led_brightness` | The brightness of the Pico Display LED, as a range from 0.0 to 1.0. Default is 0.25 (25%), 0 for off. | + +### Button Functions + +The Pico Portal has four buttons that can be used to interact with the device. The button functions are as follows: + +| Button | Function | +| :----- | :------- | +| `A` | Scroll up one line of the displayed log. Hold to scroll up faster. | +| `X` | Scroll down one line of the displayed log. Hold to scroll down faster. | +| `B` | Scroll to the top of the page of the displayed log. | +| `Y` | Scroll to the bottom of the page of the displayed log. | + +Button layout: + +```bash + Pico Display Pico Display 2.0 +|=============| |=====================| +| | | (B) (A) | +| (B) (A) | | ------------------- | +| ----------- | | | | | +| | | | | | | | +| | | | | | | | +| | | | | | | | +| | | | | | | | +| | | | | | | | +| ----------- | | | | | +| (Y) (X) | | ------------------- | +| (LED) | | (Y) (LED) (X) | +|=============| |=====================| +``` + +

[ Index ]

+ + + + + +## Development + +### Requirements + +Make sure the following are installed on your system before you begin: + +- [Node.js][url-node-js] +- [Python][url-python] +- [Thonny IDE][url-thonny-ide] + +### Development Setup + +Using a terminal, follow these steps to set up the development environment: + +1. Fork and clone the repository: + + ```bash + git clone + ``` + +2. Install project dependencies. This will install the required Node.js packages for running `setup.ts` which will download the required asset files to the `src/modules` folder. Run in the root of the project: + + ```bash + npm install + ``` + +3. Run the python setup script. This will download the files for linting (flake8), formatting (black), and pre-commit hooks (pre-commit). Basically everything we need for enforcing code quality. + + ```bash + # Install the required packages + npm run lint:install + # Install the pre-commit hooks + pre-commit install + # Update the pre-commit hooks + pre-commit autoupdate + ``` + +4. Program, test, and debug the project using the Thonny IDE. + +5. Commit (pre commit hooks should run and verify the code) and push your changes. + +6. Create a pull request [here][url-pull-requests]. + +Thank you for contributing! + +### Scripts + +| Script | Description | +| :----- | :---------- | +| `format` | Formats the Python code using [Black][url-black]. | +| `lint` | Lints the Python code using Flake8. | +| `lint:install` | Installs the required Python packages for linting and formatting. | +| `postinstall` | Downloads the required asset files to the `src/modules` folder. | + +

[ Index ]

+ + + + + +## Licensing + +This project is licensed under the **Creative Commons Attribution-NonCommercial 4.0 International (CC BY-NC 4.0)** License. See the [LICENSE.md](LICENSE.md) file for the pertaining license text. + +`SPDX-License-Identifier: CC-BY-NC-4.0` + +Basically, everything is free for you to use personally. I just ask that you don't sell this software. If you want to use it for commercial purposes, please contact me first and we can work something out. + +Feel free to build this project for yourself, your friends, or your family etc! You can also purchase this device from me directly fully assembled with a case [here](#purchase-device). + +Thank you for your understanding! + +

[ Index ]

+ + + + + +## Wrapping Up + +Thank you for all of your support, I spent a long time working on this project and plan to support it long term. Really hoping the community joins in and helps me improve it from here. It's important to me that this project stays accessible to everyone, so please keep this software free and open source. If you have any questions, please let me know by opening an issue [here][url-new-issue]. + +| Type | Info | +| :------------------------------------------------------------------------ | :------------------------------------------------------------------------ | +| | webmaster@codytolene.com | +| | https://www.buymeacoffee.com/codytolene | +| | [bc1qfx3lvspkj0q077u3gnrnxqkqwyvcku2nml86wmudy7yf2u8edmqq0a5vnt][url-btc] | + +Fin. Happy programming friend! + +Cody Tolene + + + + + + + +[img-info]: .github/images/ng-icons/info.svg +[img-pico-portal-mini-red]: .github/images/previews/mini_red.png +[img-pico-portal-xl-blue]: .github/images/previews/xl_blue.png +[img-warning]: .github/images/ng-icons/warn.svg + + + +[url-black]: https://pypi.org/project/black/ +[url-btc]: https://explorer.btc.com/btc/address/bc1qfx3lvspkj0q077u3gnrnxqkqwyvcku2nml86wmudy7yf2u8edmqq0a5vnt +[url-free-3d]: https://github.com/CodyTolene/3D-Printing +[url-lambda-guru]: https://www.lambda.guru/ +[url-new-issue]: https://github.com/CodyTolene/Pico-Portal/issues +[url-node-js]: https://nodejs.org/ +[url-pull-requests]: https://github.com/CodyTolene/Pico-Portal/pulls +[url-python]: https://www.python.org/ +[url-thingiverse]: https://www.thingiverse.com/ +[url-thonny-ide]: https://thonny.org/ + + + + diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..48563be --- /dev/null +++ b/package-lock.json @@ -0,0 +1,742 @@ +{ + "name": "pico-portal", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "pico-portal", + "version": "1.0.0", + "hasInstallScript": true, + "license": "CC-BY-NC-4.0", + "devDependencies": { + "@types/adm-zip": "^0.5.5", + "@types/node": "^22.5.1", + "adm-zip": "^0.5.16", + "rimraf": "^6.0.1", + "ts-node": "^10.9.2", + "typescript": "^5.5.4" + }, + "engines": { + "node": "^20.13.1", + "npm": ">=10.5.2 <11" + } + }, + "node_modules/@cspotcode/source-map-support": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz", + "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==", + "dev": true, + "dependencies": { + "@jridgewell/trace-mapping": "0.3.9" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@isaacs/cliui": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", + "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", + "dev": true, + "dependencies": { + "string-width": "^5.1.2", + "string-width-cjs": "npm:string-width@^4.2.0", + "strip-ansi": "^7.0.1", + "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", + "wrap-ansi": "^8.1.0", + "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", + "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", + "dev": true + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", + "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "^3.0.3", + "@jridgewell/sourcemap-codec": "^1.4.10" + } + }, + "node_modules/@pkgjs/parseargs": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", + "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", + "dev": true, + "optional": true, + "engines": { + "node": ">=14" + } + }, + "node_modules/@tsconfig/node10": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.11.tgz", + "integrity": "sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==", + "dev": true + }, + "node_modules/@tsconfig/node12": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz", + "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==", + "dev": true + }, + "node_modules/@tsconfig/node14": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz", + "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==", + "dev": true + }, + "node_modules/@tsconfig/node16": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.4.tgz", + "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==", + "dev": true + }, + "node_modules/@types/adm-zip": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/@types/adm-zip/-/adm-zip-0.5.5.tgz", + "integrity": "sha512-YCGstVMjc4LTY5uK9/obvxBya93axZOVOyf2GSUulADzmLhYE45u2nAssCs/fWBs1Ifq5Vat75JTPwd5XZoPJw==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/node": { + "version": "22.5.4", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.5.4.tgz", + "integrity": "sha512-FDuKUJQm/ju9fT/SeX/6+gBzoPzlVCzfzmGkwKvRHQVxi4BntVbyIwf6a4Xn62mrvndLiml6z/UBXIdEVjQLXg==", + "dev": true, + "dependencies": { + "undici-types": "~6.19.2" + } + }, + "node_modules/acorn": { + "version": "8.12.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.1.tgz", + "integrity": "sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-walk": { + "version": "8.3.3", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.3.tgz", + "integrity": "sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==", + "dev": true, + "dependencies": { + "acorn": "^8.11.0" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/adm-zip": { + "version": "0.5.16", + "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.5.16.tgz", + "integrity": "sha512-TGw5yVi4saajsSEgz25grObGHEUaDrniwvA2qwSC060KfqGPdglhvPMA2lPIoxs3PQIItj2iag35fONcQqgUaQ==", + "dev": true, + "engines": { + "node": ">=12.0" + } + }, + "node_modules/ansi-regex": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz", + "integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==", + "dev": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-regex?sponsor=1" + } + }, + "node_modules/ansi-styles": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz", + "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==", + "dev": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true, + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/eastasianwidth": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", + "dev": true + }, + "node_modules/emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", + "dev": true + }, + "node_modules/foreground-child": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.0.tgz", + "integrity": "sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==", + "dev": true, + "dependencies": { + "cross-spawn": "^7.0.0", + "signal-exit": "^4.0.1" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-11.0.0.tgz", + "integrity": "sha512-9UiX/Bl6J2yaBbxKoEBRm4Cipxgok8kQYcOPEhScPwebu2I0HoQOuYdIO6S3hLuWoZgpDpwQZMzTFxgpkyT76g==", + "dev": true, + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^4.0.1", + "minimatch": "^10.0.0", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^2.0.0" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/jackspeak": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-4.0.1.tgz", + "integrity": "sha512-cub8rahkh0Q/bw1+GxP7aeSe29hHHn2V4m29nnDlvCdlgU+3UGxkZp7Z53jLUdpX3jdTO0nJZUDl3xvbWc2Xog==", + "dev": true, + "dependencies": { + "@isaacs/cliui": "^8.0.2" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + }, + "optionalDependencies": { + "@pkgjs/parseargs": "^0.11.0" + } + }, + "node_modules/lru-cache": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.0.0.tgz", + "integrity": "sha512-Qv32eSV1RSCfhY3fpPE2GNZ8jgM9X7rdAfemLWqTUxwiyIC4jJ6Sy0fZ8H+oLWevO6i4/bizg7c8d8i6bxrzbA==", + "dev": true, + "engines": { + "node": "20 || >=22" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true + }, + "node_modules/minimatch": { + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.0.1.tgz", + "integrity": "sha512-ethXTt3SGGR+95gudmqJ1eNhRO7eGEGIgYA9vnPatK4/etz2MEVDno5GMCibdMTuBMyElzIlgxMna3K94XDIDQ==", + "dev": true, + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minipass": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", + "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", + "dev": true, + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/package-json-from-dist": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.0.tgz", + "integrity": "sha512-dATvCeZN/8wQsGywez1mzHtTlP22H8OEfPrVMLNr4/eGa+ijtLn/6M5f0dY8UKNrC2O9UCU6SSoG3qRKnt7STw==", + "dev": true + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-scurry": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.0.tgz", + "integrity": "sha512-ypGJsmGtdXUOeM5u93TyeIEfEhM6s+ljAhrk5vAvSx8uyY/02OvrZnA0YNGUrPXfpJMgI1ODd3nwz8Npx4O4cg==", + "dev": true, + "dependencies": { + "lru-cache": "^11.0.0", + "minipass": "^7.1.2" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rimraf": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-6.0.1.tgz", + "integrity": "sha512-9dkvaxAsk/xNXSJzMgFqqMCuFgt2+KsOFek3TMLfo8NCPfWpBmqwyNn5Y+NX56QUYfCtsyhF3ayiboEoUmJk/A==", + "dev": true, + "dependencies": { + "glob": "^11.0.0", + "package-json-from-dist": "^1.0.0" + }, + "bin": { + "rimraf": "dist/esm/bin.mjs" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/signal-exit": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", + "dev": true, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/string-width": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", + "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", + "dev": true, + "dependencies": { + "eastasianwidth": "^0.2.0", + "emoji-regex": "^9.2.2", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/string-width-cjs": { + "name": "string-width", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "node_modules/string-width-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", + "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", + "dev": true, + "dependencies": { + "ansi-regex": "^6.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/strip-ansi?sponsor=1" + } + }, + "node_modules/strip-ansi-cjs": { + "name": "strip-ansi", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ts-node": { + "version": "10.9.2", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.2.tgz", + "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", + "dev": true, + "dependencies": { + "@cspotcode/source-map-support": "^0.8.0", + "@tsconfig/node10": "^1.0.7", + "@tsconfig/node12": "^1.0.7", + "@tsconfig/node14": "^1.0.0", + "@tsconfig/node16": "^1.0.2", + "acorn": "^8.4.1", + "acorn-walk": "^8.1.1", + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "v8-compile-cache-lib": "^3.0.1", + "yn": "3.1.1" + }, + "bin": { + "ts-node": "dist/bin.js", + "ts-node-cwd": "dist/bin-cwd.js", + "ts-node-esm": "dist/bin-esm.js", + "ts-node-script": "dist/bin-script.js", + "ts-node-transpile-only": "dist/bin-transpile.js", + "ts-script": "dist/bin-script-deprecated.js" + }, + "peerDependencies": { + "@swc/core": ">=1.2.50", + "@swc/wasm": ">=1.2.50", + "@types/node": "*", + "typescript": ">=2.7" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "@swc/wasm": { + "optional": true + } + } + }, + "node_modules/typescript": { + "version": "5.5.4", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.5.4.tgz", + "integrity": "sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/undici-types": { + "version": "6.19.8", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", + "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==", + "dev": true + }, + "node_modules/v8-compile-cache-lib": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", + "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==", + "dev": true + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wrap-ansi": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", + "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", + "dev": true, + "dependencies": { + "ansi-styles": "^6.1.0", + "string-width": "^5.0.1", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs": { + "name": "wrap-ansi", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "node_modules/wrap-ansi-cjs/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true, + "engines": { + "node": ">=6" + } + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..d604e36 --- /dev/null +++ b/package.json @@ -0,0 +1,44 @@ +{ + "name": "pico-portal", + "version": "1.0.0", + "author": "Cody Tolene ", + "description": "Raspbery Pi Pico Captive Portal", + "contributors": [], + "keywords": [ + "Raspberry Pi", + "Pico", + "Captive", + "Portal" + ], + "license": "CC-BY-NC-4.0", + "homepage": "https://www.lambda.guru", + "main": "index.js", + "private": true, + "repository": { + "type": "git", + "url": "https://github.com/CodyTolene/Pico-Portal.git" + }, + "bugs": { + "url": "https://github.com/CodyTolene/Pico-Portal/issues" + }, + "engines": { + "node": "^20.13.1", + "npm": ">=10.5.2 <11" + }, + "scripts": { + "build": "tsc", + "format": "python3 -m black src/", + "lint": "python3 -m flake8 --show-source --ignore E501 src/", + "lint:install": "python3 -m pip install -r requirements.txt", + "postinstall": "ts-node setup.ts" + }, + "dependencies": {}, + "devDependencies": { + "@types/adm-zip": "^0.5.5", + "@types/node": "^22.5.1", + "adm-zip": "^0.5.16", + "rimraf": "^6.0.1", + "ts-node": "^10.9.2", + "typescript": "^5.5.4" + } +} diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..7a06125 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,3 @@ +black==24.4.2 +flake8==7.1.0 +pre-commit==3.5.0 diff --git a/setup.ts b/setup.ts new file mode 100644 index 0000000..51bf4e8 --- /dev/null +++ b/setup.ts @@ -0,0 +1,102 @@ +// ============================================================================ +// Project: Pico Portal +// License: CC-BY-NC-4.0 +// SPDX-License-Identifier: CC-BY-NC-4.0 +// Repository: https://github.com/CodyTolene/Pico-Portal +// Description: A script to download and extract specific folders from Git +// repositories. This script is used to download and extract the necessary +// files for Pico Portal MicroPython development. +// ============================================================================ + +import AdmZip from "adm-zip"; +import { sync } from "rimraf"; +import { exec } from "child_process"; +import { existsSync as pathExists, mkdirSync as makePath } from "fs"; +import { resolve as pathResolve, join as pathJoin } from "path"; + +interface Repo { + extractFromDir?: string; + extractToDir: string; + url: string; +} + +const repositories: readonly Repo[] = [ + { + extractFromDir: "phew-main/phew", + extractToDir: "./src/modules/phew", + url: "https://github.com/pimoroni/phew/archive/refs/heads/main.zip", + }, +]; + +/** + * Function to run shell commands, log the output as well. + * + * @param command - The command to run + * @returns A promise that resolves when the command is done executing + */ +function runCommand(command: string): Promise { + return new Promise((resolve, reject) => { + exec(command, (error, stdout, stderr) => { + if (error) { + console.error(`Error executing command: ${command}\n${error.message}`); + console.error(stderr); + reject(error); + } else { + resolve(); + } + }); + }); +} + +/** + * Function to download and extract specific folders from repositories. + * + * @param repos - The list of Git repositories to download and extract. + */ +async function extractRepositories(repos: readonly Repo[]): Promise { + for (const repo of repos) { + const { extractToDir, extractFromDir, url } = repo; + + const fullPath = pathResolve(extractToDir); + const zipPath = pathJoin(fullPath, "repo.zip"); + const tempExtractPath = pathResolve(extractToDir, "../temp_extract"); + + if (pathExists(fullPath)) { + sync(fullPath); + console.log(`Cleared existing files in ${fullPath}`); + makePath(fullPath, { recursive: true }); // Recreate the directory + } + + console.log(`Creating directory: ${fullPath}`); + makePath(fullPath, { recursive: true }); + + console.log(`Downloading ${url}...`); + try { + await runCommand(`curl -L ${url} -o ${zipPath}`); + + if (!pathExists(tempExtractPath)) { + makePath(tempExtractPath, { recursive: true }); + } + + const zip = new AdmZip(zipPath); + zip.extractAllTo(tempExtractPath, true); + const sourcePath = extractFromDir ? pathJoin(tempExtractPath, extractFromDir) : tempExtractPath; + + if (pathExists(sourcePath)) { + await runCommand(`mv ${sourcePath}/* ${fullPath}/`); + console.log(`Successfully copied assets to ${fullPath}`); + + sync(zipPath); + sync(tempExtractPath); + } else { + console.error(`Source path ${sourcePath} does not exist!`); + } + } catch (e) { + console.error(`Failed to process ${url}: ${e}`); + } + } + + console.log("Completed!"); +} + +extractRepositories(repositories); diff --git a/src/main.py b/src/main.py new file mode 100644 index 0000000..ea4196f --- /dev/null +++ b/src/main.py @@ -0,0 +1,61 @@ +# ============================================================================= +# Project: Pico Portal +# License: CC-BY-NC-4.0 +# Repository: https://github.com/CodyTolene/Pico-Portal +# Description: The main entry point for the Pico Portal device. This script +# will start the services for the device and keep the main loop running. +# ============================================================================= + +import uasyncio # type: ignore +import sys +import time + +# Local packages +from services.button_service import ButtonService +from services.messages_service import MessagesService +from services.onboard_led_service import OnboardLedService +from services.options_service import OptionsService +from services.pico_display_led_service import PicoDisplayLedService +from services.portal_service import PortalService + +# Ensure packages can be imported +sys.path.append("/modules") +sys.path.append("/services") + +# Version +VERSION = "1.0.0" + + +async def main(): + # Dependencies + onboard_led = OnboardLedService() + options = OptionsService() + pico_display_led = PicoDisplayLedService(options) + messages = MessagesService(options) + buttons = ButtonService(messages) + portal = PortalService(options, messages, pico_display_led) + + # Display the current version of the software on screen + await messages.display(f"Pico Portal v{VERSION}") + + # Set the display LED to white while starting up + await pico_display_led.set_color("WHITE") + + # Flash the onboard LED on and off every 3 seconds, indefinitely + # Useful for when no screen is connected to the Pico Portal + uasyncio.create_task(onboard_led.flash()) + + # Start Pico Portal services + uasyncio.create_task(portal.run()) + + # Handle the buttons and trigger actions based on button presses + uasyncio.create_task(buttons.run()) + + # Keep the application running indefinitely while the power is on + while True: + await uasyncio.sleep(1) + + +if __name__ == "__main__": + time.sleep(1) + uasyncio.run(main()) diff --git a/src/options.json b/src/options.json new file mode 100644 index 0000000..a67c1c6 --- /dev/null +++ b/src/options.json @@ -0,0 +1,8 @@ +{ + "wifi_ssid": "WiFi", + "wifi_password": "", + "wifi_domain": "setup.local", + "display_type": "DISPLAY_PICO_DISPLAY", + "enable_timestamps": false, + "led_brightness": 0.25 +} diff --git a/src/services/__init__.py b/src/services/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/services/button_service.py b/src/services/button_service.py new file mode 100644 index 0000000..4b0f7a9 --- /dev/null +++ b/src/services/button_service.py @@ -0,0 +1,125 @@ +# ============================================================================= +# Project: Pico Portal +# License: CC-BY-NC-4.0 +# Repository: https://github.com/CodyTolene/Pico-Portal +# Description: A service to handle button inputs and trigger actions based on +# the button presses. +# ============================================================================= + +from pimoroni import Button # type: ignore +import sys +import uasyncio # type: ignore + +# Local packages +from services.messages_service import MessagesService + +# Ensure packages can be imported +sys.path.append("../modules") +sys.path.append("../services") + + +class ButtonService: + def __init__(self, messages: MessagesService): + # Dependencies + self.messages = messages + + # Initialize buttons + self.button_a = Button(12) # Button A + self.button_b = Button(13) # Button B + self.button_x = Button(14) # Button X + self.button_y = Button(15) # Button Y + + # Initialize button states + self.button_states = { + "B": False, + "Y": False, + } + + async def run(self): + while True: + # Handle button presses + await self.handle_button_a() + await self.handle_button_b() + await self.handle_button_x() + await self.handle_button_y() + + # Sleep for a short period to debounce button presses + await uasyncio.sleep(0.1) + + async def handle_button_a(self): + if self.button_a.read(): + await self.scroll_continuously(self.messages.scroll_up) + else: + await uasyncio.sleep(0.1) + + async def handle_button_b(self): + if self.button_b.read(): + if not self.button_states["B"]: + self.button_states["B"] = True + self.messages.scroll_top() + else: + self.button_states["B"] = False + + async def handle_button_x(self): + if self.button_x.read(): + await self.scroll_continuously(self.messages.scroll_down) + else: + await uasyncio.sleep(0.1) + + async def handle_button_y(self): + if self.button_y.read(): + if not self.button_states["Y"]: + self.button_states["Y"] = True + self.messages.scroll_bottom() + else: + self.button_states["Y"] = False + + async def scroll_continuously(self, scroll_function): + # Continue scrolling as long as the button is pressed + while True: + scroll_function() + await uasyncio.sleep(0.2) # Adjust the scrolling speed + if not self.button_a.read() and not self.button_x.read(): + break + + +# Testing +if __name__ == "__main__": + from services.options_service import OptionsDisplayTypes, OptionsService, OptionKeys + + async def main(): + options = OptionsService() + display_type: OptionsDisplayTypes = options.get_option(OptionKeys.DISPLAY_TYPE) + + messages = MessagesService(display_type) + button_service = ButtonService(messages) + + # Start the button service + uasyncio.create_task(button_service.run()) + + # Simulate displaying messages + await messages.display("Success (green) message.", color=messages.GREEN) + await messages.display("Error (red) message!", color=messages.RED) + await messages.display("Normal (gray) message.") + await messages.display("Normal (gray) message, no timestamp.", timestamp=False) + + # Test an extra long string that has no spaces + await messages.display( + "ThisIsALongStringThatShouldBeWrappedIntoMultipleLinesBecauseItDoesNotHaveSpaces", + log=False, + ) + + # Simulate messages displaying until scrollbar appears + for i in range(20): + await messages.display(f"Message {i + 1}: Lorem ipsum dolor sit amet") + + # Start the scroll test + await messages.display("Scroll test starting...", log=False) + await uasyncio.sleep(1) + + # Keep the event loop running indefinitely to continue processing + # button inputs + while True: + await uasyncio.sleep(1) + + uasyncio.run(main()) diff --git a/src/services/messages_service.py b/src/services/messages_service.py new file mode 100644 index 0000000..9a8659c --- /dev/null +++ b/src/services/messages_service.py @@ -0,0 +1,292 @@ +# ============================================================================= +# Project: Pico Portal +# License: CC-BY-NC-4.0 +# Repository: https://github.com/CodyTolene/Pico-Portal +# Description: A service for displaying messages on the Pico Portal screen, +# with support for timestamps, colors, and scrolling. Messages are also logged +# to a log.txt file and output to the console. +# ============================================================================= + +import sys +import uasyncio # type: ignore +import utime # type: ignore +from picographics import PicoGraphics, DISPLAY_PICO_DISPLAY, DISPLAY_PICO_DISPLAY_2 # type: ignore + +# Local packages +from services.options_service import OptionsDisplayTypes, OptionKeys, OptionsService + +# Ensure packages can be imported +sys.path.append("../modules") +sys.path.append("../services") + + +class MessagesService: + def __init__(self, options: OptionsService): + display_type: OptionsDisplayTypes = options.get_option(OptionKeys.DISPLAY_TYPE) + self.enable_timestamps: bool = options.get_option(OptionKeys.ENABLE_TIMESTAMPS) + + # Initialize the display based on the display_type + if display_type == OptionsDisplayTypes.DISPLAY_PICO_DISPLAY: + self.graphics = PicoGraphics(display=DISPLAY_PICO_DISPLAY) + self.rotation = 0 # No rotation needed for DISPLAY_PICO_DISPLAY + elif display_type == OptionsDisplayTypes.DISPLAY_PICO_DISPLAY_2: + self.graphics = PicoGraphics(display=DISPLAY_PICO_DISPLAY_2, rotate=270) + self.rotation = 270 # Rotate DISPLAY_PICO_DISPLAY_2 to match orientation + else: + raise ValueError("Invalid display type") + + # Initialize drawing properties + self.BLACK = self.graphics.create_pen(0, 0, 0) + self.GRAY = self.graphics.create_pen(150, 150, 150) + self.GREEN = self.graphics.create_pen(0, 200, 0) + self.RED = self.graphics.create_pen(255, 0, 0) + self.WHITE = self.graphics.create_pen(255, 255, 255) + self.line_height = 13 + self.margin = 10 + + # Define max_lines for the selected display + self.max_lines = ( + self.graphics.get_bounds()[1] - self.margin * 2 + ) // self.line_height + + self.messages = [] + self.scroll_position = 0 + + # Use system font that supports lowercase and better character + # distinction + self.graphics.set_font("bitmap8") + + def calculate_total_lines(self): + total_lines = 0 + for msg, _ in self.messages: # Process only the message text, not the color + wrapped_lines = self.calculate_wrapped_lines(msg) + total_lines += wrapped_lines + return total_lines + + def calculate_wrapped_lines(self, message): + # Calculate the number of lines a message would take after wrapping + max_width = self.graphics.get_bounds()[0] - self.margin * 2 + wrapped_lines = 1 + current_line_length = 0 + + for word in message.split(): + word_length = self.graphics.measure_text(word, scale=1) + if current_line_length + word_length <= max_width: + current_line_length += word_length + self.graphics.measure_text( + " ", scale=1 + ) + else: + wrapped_lines += 1 + current_line_length = word_length + self.graphics.measure_text( + " ", scale=1 + ) + + return wrapped_lines + + async def display(self, message, log=True, color=None): + if self.enable_timestamps: + # Prepend the current date and time to the message in the format + # "2024-09-02 18:58:08" + current_time = utime.localtime() + formatted_time = "{:04}-{:02}-{:02} {:02}:{:02}:{:02}".format( + current_time[0], + current_time[1], + current_time[2], + current_time[3], + current_time[4], + current_time[5], + ) + # Display timestamp and message on separate lines for the screen + display_message = f"[{formatted_time}]\n{message}" + # Log timestamp and message on the same line for log.txt + log_message = f"[{formatted_time}] {message}" + else: + display_message = message + log_message = message + + # Set the color, default to GRAY if not provided + if color is None: + color = self.GRAY + + # Add the display message and its color to the list of messages + self.messages.append((f"> {display_message}", color)) + + # Calculate the total number of lines in all messages + total_lines = self.calculate_total_lines() + + # If total lines exceed the maximum, adjust the scroll position + if total_lines > self.max_lines: + self.scroll_position = total_lines - self.max_lines + + # Update the display + self.update_display(total_lines) + + if log: + print(log_message) + self.log_to_file(log_message) + + await uasyncio.sleep(1) + + def update_display(self, total_lines): + # Clear the display + self.graphics.set_pen(self.WHITE) + self.graphics.clear() + + y = self.margin + current_line = 0 + + for msg, color in self.messages: + wrapped_lines = self.calculate_wrapped_lines(msg) + if current_line + wrapped_lines > self.scroll_position: + if msg.startswith("["): + # Extract timestamp and rest of the message + timestamp, rest = msg.split("\n", 1) + self.graphics.set_pen(self.BLACK) + self.graphics.text( + timestamp, + self.margin, + y, + wordwrap=self.graphics.get_bounds()[0] - self.margin * 2, + scale=1, + ) + y += self.line_height # Move to next line after timestamp + msg = rest # Remaining message text + + self.graphics.set_pen(color) + # Split message into lines manually for better handling + lines = self.split_message_into_lines(msg) + for line in lines: + if current_line >= self.scroll_position: + self.graphics.text( + line, + self.margin, + y, + wordwrap=self.graphics.get_bounds()[0] - self.margin * 2, + scale=1, + ) + y += self.line_height + current_line += 1 + + current_line += wrapped_lines + + self.draw_scroll_bar(total_lines) + self.graphics.update() + + def split_message_into_lines(self, message): + max_width = self.graphics.get_bounds()[0] - self.margin * 2 + lines = [] + current_line = "" + + # Process the message character by character + for char in message: + if self.graphics.measure_text(current_line + char, scale=1) <= max_width: + current_line += char + else: + lines.append(current_line) + current_line = char + + if current_line: + lines.append(current_line) + + return lines + + def draw_scroll_bar(self, total_lines): + # Calculate the height and position of the scroll bar + display_height = self.graphics.get_bounds()[1] + if total_lines <= self.max_lines: + return # No need to draw a scroll bar if content fits within the screen + + scroll_bar_height = max(int(display_height * (self.max_lines / total_lines)), 5) + scrollable_area_height = display_height - scroll_bar_height + scroll_ratio = self.scroll_position / (total_lines - self.max_lines) + scroll_bar_position = int(scrollable_area_height * scroll_ratio) + + # Draw the scroll bar + self.graphics.set_pen(self.BLACK) + self.graphics.rectangle( + self.graphics.get_bounds()[0] - 5, scroll_bar_position, 5, scroll_bar_height + ) + + # Scroll up by one line + def scroll_up(self): + if self.scroll_position > 0: + self.scroll_position -= 1 + self.update_display(self.calculate_total_lines()) + + # Scroll down by one line + def scroll_down(self): + total_lines = self.calculate_total_lines() + if self.scroll_position < total_lines - self.max_lines: + self.scroll_position += 1 + self.update_display(total_lines) + + # Scroll to the top of the messages + def scroll_top(self): + self.scroll_position = 0 + self.update_display(self.calculate_total_lines()) + + # Scroll to the bottom of the messages + def scroll_bottom(self): + total_lines = self.calculate_total_lines() + if total_lines > self.max_lines: + self.scroll_position = total_lines - self.max_lines + self.update_display(total_lines) + + # Append a message to the log.txt file on a single line + def log_to_file(self, message): + try: + with open("log.txt", "a") as log_file: + log_file.write(message + "\n") + except Exception as e: + print(f"Failed to log message: {e}") + + +# Testing +if __name__ == "__main__": + + async def main(): + options = OptionsService() + + messages = MessagesService(options) + GREEN = messages.GREEN + RED = messages.RED + + # Simulate displaying messages with different colors + await messages.display("Success (green) message.", color=GREEN) + await messages.display("Error (red) message!", color=RED) + await messages.display("Normal (gray) message.") + + # Test an extra long string that has no spaces + await messages.display( + "ThisIsALongStringThatShouldBeWrappedIntoMultipleLinesBecauseItDoesNotHaveSpaces", + log=False, + ) + + # Simulate messages displaying until scrollbar appears + for i in range(20): + await messages.display(f"Message {i + 1}: Lorem ipsum dolor sit amet") + + # Start the scroll test + await messages.display("Scroll test starting...", log=False) + await uasyncio.sleep(1) + + # Scroll up x25 + for i in range(25): + messages.scroll_up() + + await uasyncio.sleep(2) + + # Scroll down x20 + for i in range(20): + messages.scroll_down() + + await uasyncio.sleep(1) + + # Scroll to top + messages.scroll_top() + await uasyncio.sleep(1) + + # Scroll to bottom + messages.scroll_bottom() + + uasyncio.run(main()) diff --git a/src/services/onboard_led_service.py b/src/services/onboard_led_service.py new file mode 100644 index 0000000..dc4772b --- /dev/null +++ b/src/services/onboard_led_service.py @@ -0,0 +1,33 @@ +# ====================================================================== +# Project: Pico Portal +# License: CC-BY-NC-4.0 +# Repository: https://github.com/CodyTolene/Pico-Portal +# Description: A service for controlling the LED lights on the Pico W. +# ====================================================================== + +import uasyncio # type: ignore +from machine import Pin # type: ignore + + +class OnboardLedService: + def __init__(self): + self.led_pin = Pin("LED", Pin.OUT) + + # Flash the LED light at a given interval + async def flash(self, interval=3.0): + while True: + self.led_pin.toggle() # Toggle on/off + await uasyncio.sleep(interval) + + +# Testing +if __name__ == "__main__": + + async def main(): + onboard_led = OnboardLedService() + uasyncio.create_task(onboard_led.flash()) + + while True: + await uasyncio.sleep(1) + + uasyncio.run(main()) diff --git a/src/services/options_service.py b/src/services/options_service.py new file mode 100644 index 0000000..179b412 --- /dev/null +++ b/src/services/options_service.py @@ -0,0 +1,67 @@ +# ============================================================================= +# Project: Pico Portal +# License: CC-BY-NC-4.0 +# Repository: https://github.com/CodyTolene/Pico-Portal +# Description: A service to handle user defined options from a JSON file. +# ============================================================================= + +import json + + +class OptionsDisplayTypes: + DISPLAY_PICO_DISPLAY = "DISPLAY_PICO_DISPLAY" + DISPLAY_PICO_DISPLAY_2 = "DISPLAY_PICO_DISPLAY_2" + + +class OptionKeys: + WIFI_SSID: str = "wifi_ssid" # Default: "WiFi" + WIFI_PASSWORD: str = "wifi_password" # Default: "" + WIFI_DOMAIN: str = "wifi_domain" # Default: "setup.local" + DISPLAY_TYPE: OptionsDisplayTypes = "display_type" # Default: DISPLAY_PICO_DISPLAY + ENABLE_TIMESTAMPS: bool = "enable_timestamps" # Default: false + LED_BRIGHTNESS = "led_brightness" # Default: 0.25 (0.0 - 1.0) + + +class OptionsService: + def __init__(self): + # Properties + self.json_file_path = "/options.json" + + # Initialization + self.options = self.load_options() + + def load_options(self): + try: + with open(self.json_file_path, "r") as f: + data = json.load(f) + return data + except OSError: + self.save_options(self.default_options()) + return self.default_options() + except ValueError: + self.save_options(self.default_options()) + return self.default_options() + + def save_options(self, options): + try: + with open(self.json_file_path, "w") as f: + json.dump(options, f) + except OSError as e: + print(f"Error writing to {self.json_file_path}: {e}") + + def get_option(self, key: OptionKeys, default=None): + return self.options.get(key, default) + + def set_option(self, key: OptionKeys, value): + self.options[key] = value + self.save_options(self.options) + + def default_options(self): + return { + OptionKeys.WIFI_SSID: "WiFi", + OptionKeys.WIFI_PASSWORD: "", + OptionKeys.WIFI_DOMAIN: "setup.local", + OptionKeys.DISPLAY_TYPE: OptionsDisplayTypes.DISPLAY_PICO_DISPLAY, + OptionKeys.ENABLE_TIMESTAMPS: False, + OptionKeys.LED_BRIGHTNESS: 0.25, + } diff --git a/src/services/pico_display_led_service.py b/src/services/pico_display_led_service.py new file mode 100644 index 0000000..b407952 --- /dev/null +++ b/src/services/pico_display_led_service.py @@ -0,0 +1,73 @@ +# ============================================================================== +# Project: Pico Portal +# License: CC-BY-NC-4.0 +# Repository: https://github.com/CodyTolene/Pico-Portal +# Description: A service for controlling the LED lights on the Pico Display(s) +# ============================================================================== + +import uasyncio # type: ignore +from pimoroni import RGBLED # type: ignore + +from services.options_service import OptionKeys, OptionsService + +COLORS = { + "RED": (255, 0, 0), + "GREEN": (0, 255, 0), + "BLUE": (0, 0, 255), + "YELLOW": (255, 255, 0), + "CYAN": (0, 255, 255), + "MAGENTA": (255, 0, 255), + "WHITE": (255, 255, 255), + "OFF": (0, 0, 0), +} + + +class PicoDisplayLedService: + def __init__(self, options: OptionsService): + # Dependencies + self.led_brightness = options.get_option(OptionKeys.LED_BRIGHTNESS) + + # Set up LED + self.led = RGBLED(6, 7, 8) # Pico display pins + + # Set the color of the LED + async def set_color(self, color: str): + if color in COLORS: + self.current_color = COLORS[color] + r, g, b = self.current_color + self.led.set_rgb(r, g, b) + await self._set_brightness(self.led_brightness) + else: + print( + f"Invalid color: {color}. Available colors are: {', '.join(COLORS.keys())}" + ) + + # Set the brightness of the current color + async def _set_brightness(self, brightness: float): + if not 0 <= brightness <= 1: + print("Invalid brightness value. It must be between 0.0 and 1.0") + return + + # Adjust current color brightness + r, g, b = self.current_color + r = int(r * brightness) + g = int(g * brightness) + b = int(b * brightness) + + # Update the LED with new brightness + self.led.set_rgb(r, g, b) + + +# Testing +if __name__ == "__main__": + + async def main(): + options = OptionsService() + led = PicoDisplayLedService(options) + + await led.set_color("GREEN") + + while True: + await uasyncio.sleep(1) + + uasyncio.run(main()) diff --git a/src/services/portal_service.py b/src/services/portal_service.py new file mode 100644 index 0000000..440bf0a --- /dev/null +++ b/src/services/portal_service.py @@ -0,0 +1,119 @@ +# ============================================================================= +# Project: Pico Portal +# License: CC-BY-NC-4.0 +# Repository: https://github.com/CodyTolene/Pico-Portal +# Description: A service to handle the portal functionality for the Pico +# Portal device. This service will start an access point, DNS server, and web +# server to allow users to connect to the device to be served web content. +# ============================================================================= + +import uasyncio # type: ignore +import sys + +# Third party packages +from modules.phew import access_point, dns, server +from modules.phew.server import redirect +from modules.phew.template import render_template + +# Local packages +from services.messages_service import MessagesService +from services.options_service import OptionKeys, OptionsService +from services.pico_display_led_service import PicoDisplayLedService + +# Ensure packages can be imported +sys.path.append("../modules") +sys.path.append("../services") + + +class PortalService: + def __init__( + self, + options: OptionsService, + messages: MessagesService, + pico_display_led: PicoDisplayLedService, + ): + # Dependencies + self.messages = messages + self.pico_display_led = pico_display_led + + # Properties + self.domain = options.get_option(OptionKeys.WIFI_DOMAIN) + self.password = options.get_option(OptionKeys.WIFI_PASSWORD) + self.ssid = options.get_option(OptionKeys.WIFI_SSID) + self.ip = None + + # Initialization + self.register_routes() + + async def start_access_point(self): + await self.messages.display("Starting access point") + ap = access_point(self.ssid, self.password) + await self.messages.display(f'AP "{self.ssid}" started') + await self.messages.display("AP Password:") + await self.messages.display(f"{self.password if self.password else 'None'}") + self.ip = ap.ifconfig()[0] + await self.messages.display("AP IP:") + await self.messages.display(self.ip) + + async def start_dns_server(self): + await self.messages.display("Starting DNS server") + await self.messages.display("Domain:") + await self.messages.display(self.domain) + + if self.ip: + dns.run_catchall(self.ip) + await self.messages.display("DNS server started") + else: + await self.messages.display("Error: Access Point not started") + raise Exception("Access Point not started") + + async def start_web_server(self): + await self.pico_display_led.set_color("GREEN") + await self.messages.display("Pico Portal started") + server.run() + + async def run(self): + try: + await self.start_access_point() + await self.start_dns_server() + await self.start_web_server() + except Exception as e: + await self.messages.display(f"Error: {e}") + + def register_routes(self): + @server.route("/", methods=["GET"]) + def index(request): + return render_template("templates/index.html") + + @server.route("/success", methods=["GET"]) + def success(request): + return render_template("templates/success.html") + + @server.route("/connecttest.txt", methods=["GET"]) + def connecttest(request): + return "", 200 + + @server.route("/ncsi.txt", methods=["GET"]) + def ncsi(request): + return "", 200 + + @server.route("/generate_204", methods=["GET"]) + def android(request): + return redirect(f"http://{self.domain}/", 302) + + @server.route("/hotspot-detect.html", methods=["GET"]) + def apple(request): + return render_template("templates/index.html") + + @server.route("/login", methods=["GET"]) + def login(request): + username = request.query.get("username") + password = request.query.get("password") + uasyncio.create_task( + self.messages.display(f"Login U: {username} P: {password}") + ) + return redirect(f"http://{self.domain}/success") + + @server.route("/", methods=["GET"]) + def catch_all(request, path): + return redirect(f"http://{self.domain}/") diff --git a/src/templates/__init__.py b/src/templates/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/src/templates/index.html b/src/templates/index.html new file mode 100644 index 0000000..13da42b --- /dev/null +++ b/src/templates/index.html @@ -0,0 +1,17 @@ + + + + + Login + + +

Login

+
+
+
+
+

+ +
+ + diff --git a/src/templates/success.html b/src/templates/success.html new file mode 100644 index 0000000..612afad --- /dev/null +++ b/src/templates/success.html @@ -0,0 +1,11 @@ + + + + + Login Success + + +

Login Successful

+

Thank you for logging in!

+ + diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..8bb6097 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,108 @@ +{ + "compilerOptions": { + /* Visit https://aka.ms/tsconfig to read more about this file */ + + /* Projects */ + // "incremental": true, /* Save .tsbuildinfo files to allow for incremental compilation of projects. */ + // "composite": true, /* Enable constraints that allow a TypeScript project to be used with project references. */ + // "tsBuildInfoFile": "./.tsbuildinfo", /* Specify the path to .tsbuildinfo incremental compilation file. */ + // "disableSourceOfProjectReferenceRedirect": true, /* Disable preferring source files instead of declaration files when referencing composite projects. */ + // "disableSolutionSearching": true, /* Opt a project out of multi-project reference checking when editing. */ + // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */ + + /* Language and Environment */ + "target": "es2016", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */ + // "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */ + // "jsx": "preserve", /* Specify what JSX code is generated. */ + // "experimentalDecorators": true, /* Enable experimental support for legacy experimental decorators. */ + // "emitDecoratorMetadata": true, /* Emit design-type metadata for decorated declarations in source files. */ + // "jsxFactory": "", /* Specify the JSX factory function used when targeting React JSX emit, e.g. 'React.createElement' or 'h'. */ + // "jsxFragmentFactory": "", /* Specify the JSX Fragment reference used for fragments when targeting React JSX emit e.g. 'React.Fragment' or 'Fragment'. */ + // "jsxImportSource": "", /* Specify module specifier used to import the JSX factory functions when using 'jsx: react-jsx*'. */ + // "reactNamespace": "", /* Specify the object invoked for 'createElement'. This only applies when targeting 'react' JSX emit. */ + // "noLib": true, /* Disable including any library files, including the default lib.d.ts. */ + // "useDefineForClassFields": true, /* Emit ECMAScript-standard-compliant class fields. */ + // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */ + + /* Modules */ + "module": "commonjs", /* Specify what module code is generated. */ + // "rootDir": "./", /* Specify the root folder within your source files. */ + // "moduleResolution": "node10", /* Specify how TypeScript looks up a file from a given module specifier. */ + // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ + // "paths": {}, /* Specify a set of entries that re-map imports to additional lookup locations. */ + // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ + // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */ + // "types": [], /* Specify type package names to be included without being referenced in a source file. */ + // "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */ + // "moduleSuffixes": [], /* List of file name suffixes to search when resolving a module. */ + // "allowImportingTsExtensions": true, /* Allow imports to include TypeScript file extensions. Requires '--moduleResolution bundler' and either '--noEmit' or '--emitDeclarationOnly' to be set. */ + // "resolvePackageJsonExports": true, /* Use the package.json 'exports' field when resolving package imports. */ + // "resolvePackageJsonImports": true, /* Use the package.json 'imports' field when resolving imports. */ + // "customConditions": [], /* Conditions to set in addition to the resolver-specific defaults when resolving imports. */ + // "resolveJsonModule": true, /* Enable importing .json files. */ + // "allowArbitraryExtensions": true, /* Enable importing files with any extension, provided a declaration file is present. */ + // "noResolve": true, /* Disallow 'import's, 'require's or ''s from expanding the number of files TypeScript should add to a project. */ + + /* JavaScript Support */ + // "allowJs": true, /* Allow JavaScript files to be a part of your program. Use the 'checkJS' option to get errors from these files. */ + // "checkJs": true, /* Enable error reporting in type-checked JavaScript files. */ + // "maxNodeModuleJsDepth": 1, /* Specify the maximum folder depth used for checking JavaScript files from 'node_modules'. Only applicable with 'allowJs'. */ + + /* Emit */ + // "declaration": true, /* Generate .d.ts files from TypeScript and JavaScript files in your project. */ + // "declarationMap": true, /* Create sourcemaps for d.ts files. */ + // "emitDeclarationOnly": true, /* Only output d.ts files and not JavaScript files. */ + // "sourceMap": true, /* Create source map files for emitted JavaScript files. */ + // "inlineSourceMap": true, /* Include sourcemap files inside the emitted JavaScript. */ + // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */ + // "outDir": "./", /* Specify an output folder for all emitted files. */ + // "removeComments": true, /* Disable emitting comments. */ + // "noEmit": true, /* Disable emitting files from a compilation. */ + // "importHelpers": true, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */ + // "downlevelIteration": true, /* Emit more compliant, but verbose and less performant JavaScript for iteration. */ + // "sourceRoot": "", /* Specify the root path for debuggers to find the reference source code. */ + // "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */ + // "inlineSources": true, /* Include source code in the sourcemaps inside the emitted JavaScript. */ + // "emitBOM": true, /* Emit a UTF-8 Byte Order Mark (BOM) in the beginning of output files. */ + // "newLine": "crlf", /* Set the newline character for emitting files. */ + // "stripInternal": true, /* Disable emitting declarations that have '@internal' in their JSDoc comments. */ + // "noEmitHelpers": true, /* Disable generating custom helper functions like '__extends' in compiled output. */ + // "noEmitOnError": true, /* Disable emitting files if any type checking errors are reported. */ + // "preserveConstEnums": true, /* Disable erasing 'const enum' declarations in generated code. */ + // "declarationDir": "./", /* Specify the output directory for generated declaration files. */ + + /* Interop Constraints */ + // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */ + // "verbatimModuleSyntax": true, /* Do not transform or elide any imports or exports not marked as type-only, ensuring they are written in the output file's format based on the 'module' setting. */ + // "isolatedDeclarations": true, /* Require sufficient annotation on exports so other tools can trivially generate declaration files. */ + // "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */ + "esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */ + // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */ + "forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */ + + /* Type Checking */ + "strict": true, /* Enable all strict type-checking options. */ + // "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */ + // "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */ + // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ + // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ + // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ + // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ + // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ + // "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ + // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ + // "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */ + // "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */ + // "noImplicitReturns": true, /* Enable error reporting for codepaths that do not explicitly return in a function. */ + // "noFallthroughCasesInSwitch": true, /* Enable error reporting for fallthrough cases in switch statements. */ + // "noUncheckedIndexedAccess": true, /* Add 'undefined' to a type when accessed using an index. */ + // "noImplicitOverride": true, /* Ensure overriding members in derived classes are marked with an override modifier. */ + // "noPropertyAccessFromIndexSignature": true, /* Enforces using indexed accessors for keys declared using an indexed type. */ + // "allowUnusedLabels": true, /* Disable error reporting for unused labels. */ + // "allowUnreachableCode": true, /* Disable error reporting for unreachable code. */ + + /* Completeness */ + // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */ + "skipLibCheck": true /* Skip type checking all .d.ts files. */ + } +}