mirror of
https://github.com/zyachel/quetre.git
synced 2025-04-03 21:17:36 +03:00
Merge branch 'zyachel:main' into mark-unanswered
This commit is contained in:
commit
6dfe488602
72 changed files with 1481 additions and 596 deletions
19
.env.example
19
.env.example
|
@ -1,11 +1,14 @@
|
|||
NODE_ENV=production #if set to development, morgan middleware will log every request
|
||||
PORT=3000 # if unset, defaults back to 3000
|
||||
CACHE_PERIOD=1h # duration for which static files' cached copies are valid in the browser(eg: 1m, 3600, '2 days'). defaults to 1h
|
||||
## if set to development, morgan middleware will log every request
|
||||
NODE_ENV=production
|
||||
|
||||
#optional properties. default shown. enable by removing # in front of them.
|
||||
### optional properties. default shown. enable by removing # in front of them.
|
||||
# PORT=3000
|
||||
## duration for which static files' cached copies are valid in the browser(eg: 1m, 3600, '2 days')
|
||||
# CACHE_PERIOD=1h
|
||||
## user agent and accept header that quora will see
|
||||
# AXIOS_USER_AGENT='axios/0.26.1'
|
||||
# AXIOS_ACCEPT='application/json, text/plain, */*'
|
||||
|
||||
#user agent and accept header that quora will see
|
||||
#AXIOS_USER_AGENT='axios/0.26.1'
|
||||
#AXIOS_ACCEPT='application/json, text/plain, */*'
|
||||
# add any value here (e.g.: 1, true, 'por favor') if you're using any service where http is the preferred method, else leave it blank
|
||||
### for specific use-cases.
|
||||
## add any value here (e.g.: 1, true, 'por favor') if you're using any service where http is the preferred method(e.g.: tor, i2p). else leave it blank
|
||||
NO_UPGRADE=
|
||||
|
|
56
CHANGELOG.md
56
CHANGELOG.md
|
@ -1,58 +1,56 @@
|
|||
# [4.0.0](https://github.com/zyachel/quetre/compare/v3.3.1...v4.0.0) (2022-09-22)
|
||||
# [5.5.0](https://github.com/zyachel/quetre/compare/v5.4.0...v5.5.0) (2023-01-15)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* fix fetcher.js ([bf266a9](https://github.com/zyachel/quetre/commit/bf266a9a8971b55400f934a1e2338e83d8fd4d38))
|
||||
|
||||
|
||||
### BREAKING CHANGES
|
||||
|
||||
* previous fetcher.js won't work as Quora again changed their HTML
|
||||
|
||||
closes https://github.com/zyachel/quetre/issues/68
|
||||
|
||||
|
||||
|
||||
## [3.3.1](https://github.com/zyachel/quetre/compare/v3.3.0...v3.3.1) (2022-09-12)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* broken layout on Tor instances ([dfec2b5](https://github.com/zyachel/quetre/commit/dfec2b5ebd0413606f64cd9f67a370aaf3d809fa))
|
||||
|
||||
|
||||
|
||||
# [3.3.0](https://github.com/zyachel/quetre/compare/v3.2.0...v3.3.0) (2022-08-03)
|
||||
* fix outgoing url on error page ([595b720](https://github.com/zyachel/quetre/commit/595b720ee12b234a9454e470139a3a40b4ad600f))
|
||||
* ui fixes for webkit-based browsers ([44229f8](https://github.com/zyachel/quetre/commit/44229f87027b1b15d38c2739b9d85bec40a36bd8))
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* add profile route ([49f5a3e](https://github.com/zyachel/quetre/commit/49f5a3e74e1c5cfd058ab2a1cc12bf5d9799a1c7))
|
||||
* add redirection route ([4199bb3](https://github.com/zyachel/quetre/commit/4199bb38c379fa5e6c2c5e58098c63534c1743b5))
|
||||
|
||||
|
||||
|
||||
# [3.2.0](https://github.com/zyachel/quetre/compare/v3.1.1...v3.2.0) (2022-07-24)
|
||||
# [5.4.0](https://github.com/zyachel/quetre/compare/v5.3.0...v5.4.0) (2023-01-07)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* fix a fatal bug in viewController.js ([33c90c1](https://github.com/zyachel/quetre/commit/33c90c17b12cf15eadde16d35fbba4cede10919b))
|
||||
* fix paragraph tag occuring inside heading tags ([65d14ba](https://github.com/zyachel/quetre/commit/65d14ba47c0d3bb1d2548972478a12a43f7e7500))
|
||||
* **routes:** add unimplemented error message to `space` route ([8820f36](https://github.com/zyachel/quetre/commit/8820f36af80f29d861a47526538293357e7c32f3))
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* add support for embedded content in answers ([bae2d7b](https://github.com/zyachel/quetre/commit/bae2d7b4f7f945d7eb55dddb4bd7e49ac21b2ae1))
|
||||
* **lang:** add ability to choose language in search route ([cca6f69](https://github.com/zyachel/quetre/commit/cca6f69deda235fa87416e28a4dd557698974e3d))
|
||||
|
||||
|
||||
|
||||
## [3.1.1](https://github.com/zyachel/quetre/compare/v3.1.0...v3.1.1) (2022-07-22)
|
||||
# [5.3.0](https://github.com/zyachel/quetre/compare/v5.2.0...v5.3.0) (2023-01-07)
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* add support for other languages ([d16ae48](https://github.com/zyachel/quetre/commit/d16ae48dcb762af6d0888b6fc556a04a4c954549))
|
||||
|
||||
|
||||
|
||||
# [5.2.0](https://github.com/zyachel/quetre/compare/v5.1.1...v5.2.0) (2022-11-27)
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* self hosted mathjax ([30d06dc](https://github.com/zyachel/quetre/commit/30d06dc0ffa1b0b362952a16ebdccc9ec2b804b9))
|
||||
|
||||
|
||||
|
||||
## [5.1.1](https://github.com/zyachel/quetre/compare/v5.1.0...v5.1.1) (2022-10-30)
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* browser theme preference not being respected when js is enabled ([40668b9](https://github.com/zyachel/quetre/commit/40668b92b5aa5c1b10cb265dc781066320cccce8))
|
||||
* fix accidental console.log statement in template ([f719b3c](https://github.com/zyachel/quetre/commit/f719b3c4c91c504db35d1077bd05aa149b0f42db))
|
||||
|
||||
|
||||
|
||||
|
|
27
CONTRIBUTING.md
Normal file
27
CONTRIBUTING.md
Normal file
|
@ -0,0 +1,27 @@
|
|||
# Contributing to Quetre
|
||||
|
||||
First of all, thanks for reading this document!
|
||||
|
||||
## Bugs and issues
|
||||
|
||||
If you happen to come across and issue or bug in Quetre, You can create a new issue by clicking [here](https://github.com/zyachel/quetre/issues/new/choose). (You can also go to the GitHub issues tab).
|
||||
|
||||
**Issues can be feature requests too!**
|
||||
|
||||
Here are some nice tips for helpful issues:
|
||||
|
||||
- **Use a good title**
|
||||
|
||||
Issue titles are important since we use them to organize the issue board, therefore, please choose titles that are short and describe the issue in detail.
|
||||
|
||||
- **Include every detail you can think of**
|
||||
|
||||
This means logs, tracebacks, use cases (if the issue is about a feature)...
|
||||
|
||||
## Pull requests and commits
|
||||
|
||||
If you wish to contribute code, you can do so by forking the repository, doing the changes and then opening a pull request for your changes to be reviewed.
|
||||
|
||||
We just please ask you to write commit messages using the [Conventional Commits standard](https://www.conventionalcommits.org/en), there is a tool called [Commitizen](https://commitizen.github.io/cz-cli/) *(or CZ)* that automatically formats and names your commit using this standard, in case you are interested.
|
||||
|
||||
About the license, by contributing to Quetre, **you agree for all your contributions to be placed under the AGPL-v3.0 license.** You can view a copy [here](https://www.gnu.org/licenses/agpl-3.0.html).
|
206
README.md
206
README.md
|
@ -1,6 +1,22 @@
|
|||
<div align="center">
|
||||
|
||||
# Quetre
|
||||
|
||||
[](https://github.com/humanetech-community/awesome-humane-tech)
|
||||
</div>
|
||||
|
||||
<div align="center">
|
||||
|
||||
[](https://github.com/humanetech-community/awesome-humane-tech)
|
||||
[](https://github.com/zyachel/quetre/issues)
|
||||

|
||||
[](https://github.com/zyachel/quetre/network)
|
||||
[](https://github.com/zyachel/quetre/stargazers)
|
||||
[](https://github.com/zyachel/quetre/blob/main/LICENSE)
|
||||

|
||||

|
||||

|
||||
|
||||
</div>
|
||||
|
||||
Quetre is an alternative front-end to Quora.
|
||||
It enables you to see answers without ads, trackers, and other such bloat.
|
||||
|
@ -9,29 +25,30 @@ It enables you to see answers without ads, trackers, and other such bloat.
|
|||
|
||||
## Key Features
|
||||
|
||||
- Privacy focused
|
||||
- **Privacy focused**
|
||||
|
||||
All requests are proxied which makes it impossible for Quora to collate meaningful data points about you.
|
||||
|
||||
- No ads or tracking
|
||||
- **No ads or tracking**
|
||||
|
||||
Absolutely no ads, no tracking, no browser fingerprinting, and no telemetry of any kind.
|
||||
|
||||
- Fully responsive layout
|
||||
- **Fully responsive layout**
|
||||
|
||||
Utilises modern CSS features like CSS Grid and Flexbox to make the website fully responsive for all screen sizes.
|
||||
|
||||
- Lightweight and fast
|
||||
- **Lightweight and fast**
|
||||
|
||||
As the website contains no bloat, pages load in a jiffy and request sizes are tiny.
|
||||
|
||||
- Dark and light themes
|
||||
- **Dark and light themes**
|
||||
|
||||
Whether you're a nightowl or bright screen lover, you'll enjoy curated color scheme for your taste.
|
||||
|
||||
- Unofficial API support
|
||||
- **Unofficial API support**
|
||||
|
||||
just add `/api/v1/` after the domain name in the URL and get a JSON response.
|
||||
|
||||
just add `/api/v1/` after the domain name in the URL and get a JSON response.
|
||||
|
||||
---
|
||||
|
||||
|
@ -48,29 +65,38 @@ It enables you to see answers without ads, trackers, and other such bloat.
|
|||
<!-- prettier-ignore -->
|
||||
| Instance | Region | Provider | Notes |
|
||||
| -------- | ------ | -------- | ----- |
|
||||
| 1. Clearnet | | | |
|
||||
| [quetre.iket.me](https://quetre.iket.me/) | Canada | OVHCloud | Official instance |
|
||||
| [quora.vern.cc](https://qr.vern.cc) | Canada | OVHCloud | Operated by [~vern](https://vern.cc/) |
|
||||
| [quetre.pussthecat.org](https://quetre.pussthecat.org/) | Germany | – | Operated by [PussTheCat.org](https://pussthecat.org/) |
|
||||
| [wuetre.herokuapp.com](https://wuetre.herokuapp.com/) | Europe | Heroku | Operated by AnonymousZ |
|
||||
| [quetreus.herokuapp.com](https://quetreus.herokuapp.com/) | U.S. | Heroku | Operated by [toyboatcash](https://github.com/toyboatcash) |
|
||||
| **1. Clearnet** | | | |
|
||||
| [quetre.iket.me](https://quetre.iket.me) | Canada | OVHCloud | Official instance |
|
||||
| [quora.vern.cc](https://qr.vern.cc) | US | Hetzner | Operated by [~vern](https://vern.cc/) |
|
||||
| [quetre.pussthecat.org](https://quetre.pussthecat.org) | Germany | – | Operated by [PussTheCat.org](https://pussthecat.org/) |
|
||||
| [quetre.tokhmi.xyz](https://quetre.tokhmi.xyz/) | U.S. | Oracle | Operated by [Tokhmi](https://tokhmi.xyz) |
|
||||
| [quetre.projectsegfau.lt](https://quetre.projectsegfau.lt/) | Europe | BuyVM | Operated by [Project Segfault](https://projectsegfau.lt) |
|
||||
| [quetre.esmailelbob.xyz](https://quetre.esmailelbob.xyz/) | Canada | OVHCloud | Operated by [Esmail EL BoB](https://esmailelbob.xyz/) |
|
||||
| [quetre.projectsegfau.lt](https://quetre.projectsegfau.lt) | Europe | BuyVM | Operated by [Project Segfault](https://projectsegfau.lt) |
|
||||
| [quetre.esmailelbob.xyz](https://quetre.esmailelbob.xyz) | Canada | OVHCloud | Operated by [Esmail EL BoB](https://esmailelbob.xyz/) |
|
||||
| [quetre.odyssey346.dev](https://quetre.odyssey346.dev) | Poland | OVHCloud | Operated by [Odyssey346](https://odyssey346.dev/) |
|
||||
| 2. Onion | | | |
|
||||
| [quetre.privacydev.net](https://quetre.privacydev.net) | U.S. | BuyVM | Operated by [PrivacyDev](https://privacydev.net/) |
|
||||
| [ask.habedieeh.re](https://ask.habedieeh.re) | Canada | Oracle | Operated by [habedieeh.re](https://www.habedieeh.re) |
|
||||
| [quetre.marcopisco.com](https://quetre.marcopisco.com) | Portugal | Vodafone Portugal (Cloudflare) | Operated by [marcopisco.com](https://www.marcopisco.com) |
|
||||
| [quetre.blackdrgn.nl](https://quetre.blackdrgn.nl) | Germany | Contabo | Operated by [blackdrgn.nl](https://blackdrgn.nl) |
|
||||
| [quetre.pufe.org](https://quetre.pufe.org) | New Zealand | - | Operated by pufe.org |
|
||||
| [quetre.lunar.icu](https://quetre.lunar.icu) | Germany | Cloudflare | Operated by [lunar.icu](https://lunar.icu/) |
|
||||
| [que.wilbvr.me](https://que.wilbvr.me) | Netherlands | Liga Hosting | Operated by [Wilbvr](https://wilbvr.me) |
|
||||
| [quora.femboy.hu](https://quora.femboy.hu) | Hungary | N/A (Self-hosted) | Operated by [hnhx](https://femboy.hu) |
|
||||
| **2. Onion** | | | |
|
||||
| [quetre.esmail5pdn24shtvieloeedh7ehz3nrwcdivnfhfcedl7gf4kwddhkqd.onion](http://quetre.esmail5pdn24shtvieloeedh7ehz3nrwcdivnfhfcedl7gf4kwddhkqd.onion) | Canada | OVHCloud | Operated by [Esmail EL BoB](https://esmailelbob.xyz/) |
|
||||
| [qr.vernccvbvyi5qhfzyqengccj7lkove6bjot2xhh5kajhwvidqafczrad.onion](http://qr.vernccvbvyi5qhfzyqengccj7lkove6bjot2xhh5kajhwvidqafczrad.onion/) | Canada | OVHCloud | Operated by [~vern](https://vern.cc) |
|
||||
| 3. I2P | | | |
|
||||
| [http://qr.vern.i2p/](http://vernnflenvsqccuanaun7yydnmturi4jkyxlyzhn6ultpje66c3q.b32.i2p/) | Canada | OVHCloud | Operated by [~vern](https://vern.cc) |
|
||||
| [qr.vernccvbvyi5qhfzyqengccj7lkove6bjot2xhh5kajhwvidqafczrad.onion](http://qr.vernccvbvyi5qhfzyqengccj7lkove6bjot2xhh5kajhwvidqafczrad.onion/) | US | Hetzner | Operated by [~vern](https://vern.cc) |
|
||||
| [ask.habeehrhadazsw3izbrbilqajalfyqqln54mrja3iwpqxgcuxnus7eid.onion](http://ask.habeehrhadazsw3izbrbilqajalfyqqln54mrja3iwpqxgcuxnus7eid.onion/) | Canada | Oracle | Operated by [habedieeh.re](https://www.habedieeh.re) |
|
||||
| [quetre.g4c3eya4clenolymqbpgwz3q3tawoxw56yhzk4vugqrl6dtu3ejvhjid.onion](http://quetre.g4c3eya4clenolymqbpgwz3q3tawoxw56yhzk4vugqrl6dtu3ejvhjid.onion/) | U.S. | BuyVM | Operated by [PrivacyDev](https://privacydev.net/) |
|
||||
| [quora.cepyxplublbyw2f4axy4pyztfbxmf63lrt2c7uwv6wl4iixz53czload.onion](http://quora.cepyxplublbyw2f4axy4pyztfbxmf63lrt2c7uwv6wl4iixz53czload.onion) | Hungary | N/A (Self-hosted) | Operated by [hnhx](https://femboy.hu) |
|
||||
| **3. I2P** | | | |
|
||||
| [qr.vern.i2p/](http://vernnflenvsqccuanaun7yydnmturi4jkyxlyzhn6ultpje66c3q.b32.i2p/) | US | Hetzner | Operated by [~vern](https://vern.cc) |
|
||||
|
||||
---
|
||||
|
||||
## Comparision
|
||||
## Comparison
|
||||
|
||||
### Speed
|
||||
|
||||
URL for comparision: https://www.quora.com/How-does-the-Z-boson-decay
|
||||
URL for comparison: https://www.quora.com/How-does-the-Z-boson-decay
|
||||
|
||||
| | Quora | Quetre |
|
||||
| --------------- | --------- | -------- |
|
||||
|
@ -79,13 +105,13 @@ URL for comparision: https://www.quora.com/How-does-the-Z-boson-decay
|
|||
| Finish time | 2.44min\* | 4.62s |
|
||||
| Data consumed | 3.49MB | 404.47KB |
|
||||
|
||||
\*the requests were ongoing even after 6 minutes
|
||||
\*The requests were ongoing even after 6 minutes
|
||||
|
||||
---
|
||||
|
||||
### Usability
|
||||
|
||||
- Quora: You can't even see an answer(unless you do some hacks) if you're not signed in. They put a big banner in front of answers to sign you up/in forcefully.
|
||||
- Quora: You can't even see an answer (unless you do some hacks) if you're not signed in. They put a big banner in front of answers to sign you up/in forcefully.
|
||||
|
||||
- Quetre: There is no accounts system. Just read whatever you want to read. Zero fuss.
|
||||
|
||||
|
@ -93,92 +119,84 @@ URL for comparision: https://www.quora.com/How-does-the-Z-boson-decay
|
|||
|
||||
### Privacy
|
||||
|
||||
#### Quora(when browsing anonymously)
|
||||
#### Quora (when browsing "anonymously")
|
||||
|
||||
From [their privacy policy](https://www.quora.com/about/privacy)
|
||||
|
||||
- Technologies used
|
||||
- cookies
|
||||
- log files
|
||||
- clear GIFs/pixel tags
|
||||
- **Technologies used**
|
||||
- Cookies
|
||||
- Log files
|
||||
- Clear GIFs/pixel tags
|
||||
- JavaScript
|
||||
- web beacons
|
||||
- local storage objects
|
||||
- Web beacons
|
||||
- Local Storage Objects
|
||||
- Analytics Tools
|
||||
- other tracking technologies
|
||||
- Data collected
|
||||
- searches
|
||||
- page views
|
||||
- date and time of your visit
|
||||
- browser type
|
||||
- type of computer or mobile device
|
||||
- browser language
|
||||
- Other tracking technologies
|
||||
- **Data collected**
|
||||
- Searches
|
||||
- Page views
|
||||
- Date and time of your visit
|
||||
- Browser type
|
||||
- Type of computer or mobile device
|
||||
- Browser language
|
||||
- IP address
|
||||
- mobile carrier
|
||||
- unique device identifier
|
||||
- location
|
||||
- requested and referring URLs
|
||||
- other information about your use of the Quora Platform
|
||||
- Mobile carrier
|
||||
- Unique device identifier
|
||||
- Location
|
||||
- Requested and referring URLs
|
||||
- Other information about your use of the Quora Platform
|
||||
|
||||
#### Quetre
|
||||
|
||||
- Data actively collected by Quetre
|
||||
- **Data actively collected by Quetre**
|
||||
|
||||
None.
|
||||
|
||||
- Data passively collected by Quetre
|
||||
- **Data passively collected by Quetre**
|
||||
|
||||
Whenever you hit some error page, an error object is logged to the console on the server. That error object contains the resource url you were trying to access, and the usual stack trace. That's it.
|
||||
Whenever you hit some error page, an error object is logged to the console on the server. That error object contains the resource URL you were trying to access, and the usual stack trace. That's it.
|
||||
|
||||
- Data stored locally in your browser
|
||||
- **Data stored locally in your browser**
|
||||
|
||||
A key called 'theme' is stored in local storage provided by your browser to store your theme preference should you override the default theme. To prevent this behaviour, either disable JavaScript or local storage for Quetre.
|
||||
|
||||
- Data collected by other services
|
||||
|
||||
If you're using the official instance(which is deployed on Heroku), Heroku might log your IP to prevent abuse. Also, as Quetre connects to 'cdn.jsdelivr.net' for MathJax library respectively, this services might log some data. So, follow due precaution. Using a VPN might be a good idea. Or even better, consider hosting your own instance.
|
||||
|
||||
---
|
||||
|
||||
## FAQs
|
||||
|
||||
- How do I use this?
|
||||
- **How do I use this?**
|
||||
|
||||
Replace 'www.quora.com' in any URL with 'quetre.iket.me'. So, 'https://www.quora.com/Are-Nubians-nilotes' becomes 'https://quetre.iket.me/Are-Nubians-nilotes'.
|
||||
Replace 'www.quora.com' in any URL with 'quetre.iket.me' (or any other instance). So, 'https://www.quora.com/Are-Nubians-nilotes' becomes 'https://quetre.iket.me/Are-Nubians-nilotes'.
|
||||
|
||||
- I don't want to edit the URLs manually!
|
||||
- **I don't want to edit the URLs manually!**
|
||||
|
||||
There are [a couple of solutions](#automatic-redirection) for that.
|
||||
|
||||
- There are some unreachable routes.
|
||||
- **There are some unreachable routes.**
|
||||
|
||||
I'm working to implement them soon. Keep an eye on [To-Do list](#to-do).
|
||||
|
||||
- Why is the website connecting to 'cdn.jsdelivr.net'?
|
||||
|
||||
It is for an open source library – [Mathjax](https://www.mathjax.org/) – which is used to display math eqations nicely. If I get enough time, I'll include it locally.
|
||||
|
||||
- Why are some math equations showing up weirdly?
|
||||
- **Why are some math equations showing up weirdly?**
|
||||
|
||||
If you're browsing with JavaScript disabled, then the Mathjax library isn't able to load and format tex equations. I'd recommend to enable JavaScript for it since there's no other way to show them in the browser. Even Quora uses Mathjax.
|
||||
|
||||
- Why can I only view a couple of answers?
|
||||
- **Why can I only view a couple of answers?**
|
||||
|
||||
Quora doesn't show all answers at once. It only loads more answers as the user scrolls down. Furthermore, it uses many unique IDs to send ajax requests to fetch those answers. So, all in all, getting more answers isn't impossible but quite difficult requiring some serious amount of time on their website in order to figure out how it all happens. I'm short on time for now.
|
||||
|
||||
- Why am I getting a _Recheck the URL_ error?
|
||||
- **Why am I getting a _Recheck the URL_ error?**
|
||||
|
||||
Sometimes Quora doesn't populate the answer page HTML, and hence, Quetre is unable to extract data from it. If that happens, you can refresh the page a couple of times to get the answers.
|
||||
|
||||
- I have some ideas/want to help.
|
||||
- **I have some ideas/want to help.**
|
||||
|
||||
You're most welcome to do that. Just [contact me](#contact) or fork [the repo](https://github.com/zyachel/quetre/fork) and make a pull request. You can even help by correcting some typos or translating this README to other languages.
|
||||
You're most welcome to do that. Just [contact me](#contact) or fork [the repo](https://github.com/zyachel/quetre/fork) and make a pull request. You can even help by correcting some typos or translating this README to other languages. *Please, read the [CONTRIBUTING.md](./CONTRIBUTING.md) beforehand.*
|
||||
|
||||
- Why the name Quetre?
|
||||
- **Why the name Quetre?**
|
||||
|
||||
Quora is [supposedly](https://www.quora.com/Why-is-Quora-called-Quora-4) a portmanteau of 'Questions or answers'. In the same vein, Quetre is a portmanteau of 'Questions and answers', but [in Latin](https://lingva.ml/en/la/questions%20and%20answers%0A).
|
||||
Quora is [supposedly](https://www.quora.com/Why-is-Quora-called-Quora-4) a portmanteau of 'Questions or answers'. In the same vein, Quetre is a portmanteau of 'Questions and answers', but [in Latin](https://lingva.ml/en/la/questions%20and%20answers%0A).
|
||||
|
||||
- I cannot view the comments. Will you add that feature?
|
||||
- **I cannot view the comments. Will you add that feature?**
|
||||
|
||||
See [this issue](https://codeberg.org/zyachel/quetre/issues/11)
|
||||
|
||||
|
@ -186,12 +204,12 @@ From [their privacy policy](https://www.quora.com/about/privacy)
|
|||
|
||||
## To-Do
|
||||
|
||||
- [ ] add missing routes like topics, profile, and search
|
||||
- [ ] use redis
|
||||
- [ ] serve images and other assets from Quetre
|
||||
- [x] implement a better installation method
|
||||
- [ ] implement other trivial routes like a specific answer, spaces, etc.
|
||||
- [ ] implement a way to get more answers(not a big priority as of now)
|
||||
- [x] Add missing routes like topics, profile, and search
|
||||
- [ ] Use redis
|
||||
- [x] Serve images and other assets from Quetre
|
||||
- [x] Implement a better installation method
|
||||
- [ ] Implement other trivial routes like a specific answer, spaces, etc.
|
||||
- [ ] Implement a way to get more answers (not a big priority as of now)
|
||||
|
||||
---
|
||||
|
||||
|
@ -204,47 +222,65 @@ From [their privacy policy](https://www.quora.com/about/privacy)
|
|||
2. Clone and set up the repository.
|
||||
|
||||
```bash
|
||||
git clone https://github.com/zyachel/quetre.git # replace github.com with codeberg.org if you're cloning from there
|
||||
git clone https://github.com/zyachel/quetre.git # Replace github.com with codeberg.org if you're cloning from there
|
||||
cd quetre
|
||||
cp .env.example .env # you can make any changes here
|
||||
# change `pnpm` to `npm run` here as well as in package.json if you use `npm`
|
||||
cp .env.example .env # You can make any changes here
|
||||
# Change `pnpm` to `npm run` here as well as in package.json if you use `npm`
|
||||
pnpm install
|
||||
pnpm start
|
||||
```
|
||||
|
||||
Quetre will start running at http://localhost:3000.
|
||||
|
||||
**Important: If you are running an .onion or I2P instance, set the `NO_UPGRADE` .env variable to "1", otherwise the page will be broken.**
|
||||
|
||||
### Docker
|
||||
|
||||
There is a [docker image](https://github.com/PussTheCat-org/docker-quetre-quay) made by [@TheFrenchGhosty](https://github.com/TheFrenchGhosty) for [PussTheCat.org](https://pussthecat.org/)'s [instance](https://quetre.pussthecat.org/). You can check that out case you want to go the docker way.
|
||||
There is a [docker image](https://github.com/PussTheCat-org/docker-quetre-quay) made by [@TheFrenchGhosty](https://github.com/TheFrenchGhosty) for [PussTheCat.org](https://pussthecat.org/)'s [instance](https://quetre.pussthecat.org/).
|
||||
If you want a leaner one, you can checkout [@video-prize-ranch](https://codeberg.org/video-prize-ranch)'s [docker image](https://codeberg.org/video-prize-ranch/-/packages/container/quetre/latest).
|
||||
|
||||
---
|
||||
|
||||
## Contributing
|
||||
|
||||
The development may seem slow as I don't have lots of free time. And whenever I do, it gets split between this service and [LibreIMDb](https://github.com/zyachel/libremdb/).
|
||||
If you believe you can help furthering this project in any way (be it maintaining, fixing issues, or adding features), please [get in touch](#contact).
|
||||
*Regardless, any type of contribution is always welcome.*
|
||||
|
||||
**If you plan on contributing, please read our [CONTRIBUTING.md](./CONTRIBUTING.md) file.**
|
||||
|
||||
## Misc
|
||||
|
||||
### Automatic redirection
|
||||
|
||||
Following extensions can be used to automatically redirect Quora URLs to Quetre:
|
||||
The following extensions can be used to automatically redirect Quora URLs to Quetre:
|
||||
|
||||
- [redirector](https://github.com/einaregilsson/Redirector)
|
||||
You can manually add any redirect.
|
||||
Below is a basic config of Quora to Quetre. Replace `quetre.herokuapp.com` in `Redirect to` to any instance of your choice.
|
||||
Below is a basic config of Quora to Quetre. Replace `quetre.iket.me` in `Redirect to` to any instance of your choice.
|
||||
|
||||
```
|
||||
Description: Quora to Quetre
|
||||
Example URL: https://www.quora.com/What-is-Linux-4?share=1
|
||||
Include pattern: https?:\/\/(www\.)?quora\.com\/([^\?]*)
|
||||
Redirect to: https://quetre.herokuapp.com/$2
|
||||
Include pattern: (https:\/\/.{2,}\.quora\.com\/.*)
|
||||
Redirect to: https://quetre.iket.me/redirect/$1
|
||||
Pattern type: Regular Expression
|
||||
Pattern description: redirects all Quora urls(excluding language-specific and spaces) to Quetre
|
||||
Pattern description: Redirects all Quora urls to Quetre
|
||||
```
|
||||
|
||||
This config should output:
|
||||
`Example result: https://quetre.herokuapp.com/What-is-Linux-4`
|
||||
`Example result: https://quetre.iket.me/redirect/https://www.quora.com/What-is-Linux-4?share=1`
|
||||
|
||||
- [LibRedirect](https://github.com/libredirect/libredirect/)
|
||||
Redirects many popular services to their alternative front-ends. Has a ton of features and an active community. Quetre is supported by default. So, no need to do anything.
|
||||
|
||||
- [Privacy Redirector](https://github.com/dybdeskarphet/privacy-redirector)
|
||||
A userscript that redirects popular social media platforms to their privacy respecting frontends.
|
||||
|
||||
- Other addons with similar functionality:
|
||||
- [Dynamic Privacy Redirect](https://github.com/PrivacyDevel/DPR-addon)
|
||||
- [Alter](https://github.com/w3bdev1/alter)
|
||||
|
||||
### Other alternative front-ends
|
||||
|
||||
- [digitalblossom/alternative-frontends](https://github.com/digitalblossom/alternative-frontends): contains other alternative front-ends.
|
||||
|
@ -256,7 +292,7 @@ Following extensions can be used to automatically redirect Quora URLs to Quetre:
|
|||
|
||||
### Programming
|
||||
|
||||
- [JavaScript](https://www.ecma-international.org/technical-committees/tc39/): programming language
|
||||
- [JavaScript](https://www.ecma-international.org/technical-committees/tc39/): Programming language
|
||||
- [Sass](https://sass-lang.com/): CSS preprocessor
|
||||
- [Pug](https://pugjs.org/): Template engine
|
||||
- [Node.js](https://nodejs.org/en/): JS runtime environment
|
||||
|
|
7
app.js
7
app.js
|
@ -20,10 +20,9 @@ const app = express();
|
|||
// 1. IMPORTANT MIDDLWARES
|
||||
app.use(compression()); // compressing responses
|
||||
app.use(
|
||||
helmet({
|
||||
helmet({
|
||||
contentSecurityPolicy: {
|
||||
directives: {
|
||||
'script-src': ["'self'", 'cdn.jsdelivr.net'],
|
||||
'block-all-mixed-content': null, // deprecated.
|
||||
'upgrade-insecure-requests': process.env.NO_UPGRADE ? null : [],
|
||||
},
|
||||
|
@ -51,9 +50,7 @@ app.use(
|
|||
if (process.env.NODE_ENV === 'development') app.use(morgan('dev')); // for logging during development
|
||||
// middleware to add baseUrl to req object
|
||||
app.use((req, res, next) => {
|
||||
req.urlObj = new URL(
|
||||
`${req.protocol}://${req.get('host')}${req.originalUrl}`
|
||||
);
|
||||
req.urlObj = new URL(req.originalUrl, `${req.protocol}://${req.get('host')}`);
|
||||
next();
|
||||
});
|
||||
|
||||
|
|
8
app.json
8
app.json
|
@ -1,8 +0,0 @@
|
|||
{
|
||||
"name": "Quetre",
|
||||
"description": "A libre front-end for Quora.",
|
||||
"keywords": ["front-end", "quora", "privacy", "foss"],
|
||||
"website": "https://quetre.herokuapp.com/",
|
||||
"repository": "https://github.com/zyachel/quetre",
|
||||
"logo": "https://quetre.herokuapp.com/icon.svg"
|
||||
}
|
|
@ -1,11 +1,13 @@
|
|||
/* eslint-disable no-unused-vars */
|
||||
////////////////////////////////////////////////////////
|
||||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
import axiosInstance from '../utils/axiosInstance.js';
|
||||
import getAxiosInstance from '../utils/getAxiosInstance.js';
|
||||
import catchAsyncErrors from '../utils/catchAsyncErrors.js';
|
||||
import getAnswers from '../fetchers/getAnswers.js';
|
||||
import getTopic from '../fetchers/getTopic.js';
|
||||
import getProfile from '../fetchers/getProfile.js';
|
||||
import getSearch from '../fetchers/getSearch.js';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// EXPORTS
|
||||
|
@ -13,18 +15,24 @@ import getProfile from '../fetchers/getProfile.js';
|
|||
export const about = (req, res, next) => {
|
||||
res.status(200).json({
|
||||
status: 'success',
|
||||
message:
|
||||
"make a request. available endpoints are: '/some-slug', '/unanswered/some-slug'",
|
||||
message: `make a request.
|
||||
available endpoints are: '/slug', '/unanswered/slug', '/topic/slug', '/profile/slug', '/search?q=query', /?q=query.`,
|
||||
});
|
||||
};
|
||||
|
||||
export const answers = catchAsyncErrors(async (req, res, next) => {
|
||||
const data = await getAnswers(req.params.slug);
|
||||
const { slug } = req.params;
|
||||
const { lang } = req.query;
|
||||
|
||||
const data = await getAnswers(slug, lang);
|
||||
res.status(200).json({ status: 'success', data });
|
||||
});
|
||||
|
||||
export const topic = catchAsyncErrors(async (req, res, next) => {
|
||||
const data = await getTopic(req.params.slug);
|
||||
const { slug } = req.params;
|
||||
const { lang } = req.query;
|
||||
|
||||
const data = await getTopic(slug, lang);
|
||||
res.status(200).json({ status: 'success', data });
|
||||
});
|
||||
|
||||
|
@ -33,6 +41,15 @@ export const profile = catchAsyncErrors(async (req, res, next) => {
|
|||
res.status(200).json({ status: 'success', data });
|
||||
});
|
||||
|
||||
export const search = catchAsyncErrors(async (req, res, next) => {
|
||||
const searchText = req.urlObj.searchParams.get('q')?.trim(); // no search to perform if there isn't any query
|
||||
|
||||
let searchData = null;
|
||||
if (searchText) searchData = await getSearch(req.urlObj.search);
|
||||
|
||||
res.status(200).json({ status: 'success', data: searchData });
|
||||
});
|
||||
|
||||
export const unimplemented = (req, res, next) => {
|
||||
res.status(501).json({
|
||||
status: 'fail',
|
||||
|
@ -41,17 +58,19 @@ export const unimplemented = (req, res, next) => {
|
|||
};
|
||||
|
||||
export const image = catchAsyncErrors(async (req, res, next) => {
|
||||
if (!req.params.domain.endsWith('quoracdn.net')) {
|
||||
const { domain, path } = req.params;
|
||||
if (!domain.endsWith('quoracdn.net')) {
|
||||
return res.status(403).json({
|
||||
status: 'fail',
|
||||
message: 'Invalid domain',
|
||||
});
|
||||
}
|
||||
// changing defaults for this particular endpoint
|
||||
const axiosInstance = getAxiosInstance();
|
||||
axiosInstance.defaults.baseURL = `https://${domain}/`;
|
||||
|
||||
const imageRes = await axiosInstance.get(path, { responseType: 'stream' });
|
||||
|
||||
const imageRes = await axiosInstance.get(
|
||||
`https://${req.params.domain}/${req.params.path}`,
|
||||
{ responseType: 'arraybuffer' }
|
||||
);
|
||||
res.set('Content-Type', imageRes.headers['content-type']);
|
||||
res.status(200).send(imageRes.data);
|
||||
return imageRes.data.pipe(res);
|
||||
});
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
/* eslint-disable no-unused-vars */
|
||||
/* eslint-disable no-param-reassign */
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
|
@ -34,9 +35,8 @@ const sendErrorResponse = (err, req, res, devMode = false) => {
|
|||
},
|
||||
meta: {
|
||||
title: 'Error',
|
||||
url: `${req.urlObj.origin}${req.urlObj.pathname}`,
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
urlObj: req.urlObj,
|
||||
description: `ERROR: ${err.message}. Please try again later.`,
|
||||
},
|
||||
});
|
||||
|
|
|
@ -5,8 +5,9 @@
|
|||
import catchAsyncErrors from '../utils/catchAsyncErrors.js';
|
||||
import getAnswers from '../fetchers/getAnswers.js';
|
||||
import getTopic from '../fetchers/getTopic.js';
|
||||
import { nonSlugRoutes } from '../utils/constants.js';
|
||||
import { acceptedLanguages, nonSlugRoutes } from '../utils/constants.js';
|
||||
import getProfile from '../fetchers/getProfile.js';
|
||||
import getSearch from '../fetchers/getSearch.js';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// EXPORTS
|
||||
|
@ -15,7 +16,7 @@ export const about = (req, res, next) => {
|
|||
res.render('about', {
|
||||
meta: {
|
||||
title: 'About',
|
||||
url: `${req.urlObj.origin}${req.urlObj.pathname}`,
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
description:
|
||||
'Quetre is a libre front-end for Quora. See any answer without being tracked, without being required to log in, and without being bombarded by pesky ads.',
|
||||
|
@ -27,7 +28,7 @@ export const privacy = (req, res, next) => {
|
|||
res.render('privacy', {
|
||||
meta: {
|
||||
title: 'Privacy',
|
||||
url: `${req.urlObj.origin}${req.urlObj.pathname}`,
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
description: 'Privacy Policy of Quetre, a libre front-end for Quora.',
|
||||
},
|
||||
|
@ -36,10 +37,12 @@ export const privacy = (req, res, next) => {
|
|||
|
||||
export const answers = catchAsyncErrors(async (req, res, next) => {
|
||||
const { slug } = req.params;
|
||||
const { lang } = req.query;
|
||||
|
||||
// added this so that a request by browser to get favicon doesn't end up being interpreted as a slug
|
||||
if (nonSlugRoutes.includes(slug)) return next();
|
||||
|
||||
const answersData = await getAnswers(slug);
|
||||
const answersData = await getAnswers(slug, lang);
|
||||
const title = answersData.question.text[0].spans
|
||||
.map(span => span.text)
|
||||
.join('');
|
||||
|
@ -48,7 +51,7 @@ export const answers = catchAsyncErrors(async (req, res, next) => {
|
|||
data: answersData,
|
||||
meta: {
|
||||
title,
|
||||
url: `${req.urlObj.origin}${req.urlObj.pathname}`,
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
description: `Answers to ${title}`,
|
||||
},
|
||||
|
@ -56,13 +59,16 @@ export const answers = catchAsyncErrors(async (req, res, next) => {
|
|||
});
|
||||
|
||||
export const topic = catchAsyncErrors(async (req, res, next) => {
|
||||
const topicData = await getTopic(req.params.slug);
|
||||
const { slug } = req.params;
|
||||
const { lang } = req.query;
|
||||
|
||||
const topicData = await getTopic(slug, lang);
|
||||
|
||||
res.status(200).render('topic', {
|
||||
data: topicData,
|
||||
meta: {
|
||||
title: topicData.name,
|
||||
url: `${req.urlObj.origin}${req.urlObj.pathname}`,
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
description: `Information about ${topicData.name} topic.`,
|
||||
},
|
||||
|
@ -70,33 +76,70 @@ export const topic = catchAsyncErrors(async (req, res, next) => {
|
|||
});
|
||||
|
||||
export const profile = catchAsyncErrors(async (req, res, next) => {
|
||||
const profileData = await getProfile(req.params.name);
|
||||
const { name } = req.params;
|
||||
const { lang } = req.query;
|
||||
const profileData = await getProfile(name, lang);
|
||||
|
||||
res.status(200).render('profile', {
|
||||
data: profileData,
|
||||
meta: {
|
||||
title: profileData.basic.name,
|
||||
url: `${req.urlObj.origin}${req.urlObj.pathname}`,
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
description: `${profileData.basic.name}'s profile.`,
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
export const unimplemented = (req, res, next) => {
|
||||
const message =
|
||||
"This route isn't yet implemented. Check back sometime later!";
|
||||
res.status(501).render('error', {
|
||||
data: {
|
||||
statusCode: 501,
|
||||
message,
|
||||
export const search = catchAsyncErrors(async (req, res, next) => {
|
||||
const searchText = req.urlObj.searchParams.get('q')?.trim();
|
||||
const { lang } = req.query;
|
||||
let searchData = null;
|
||||
if (searchText) searchData = await getSearch(req.urlObj.search, lang);
|
||||
|
||||
res.status(200).render('search', {
|
||||
data: searchData,
|
||||
meta: {
|
||||
title: searchText || 'Search',
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
description: searchText ? `results for '${searchText}'` : 'search page',
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
const regex = /^https:\/\/(.{2,})\.quora\.com(\/.*)$/; // local helper constant
|
||||
export const redirect = (req, res, next) => {
|
||||
const url = req.originalUrl.replace('/redirect/', ''); // removing `/redirect/` part.
|
||||
const match = regex.exec(url);
|
||||
|
||||
if (!match) return res.redirect('/');
|
||||
|
||||
const [_, subdomain, rest] = match; // eg: subdomain: 'es', rest: '/topic/linux?share=1'
|
||||
let link;
|
||||
|
||||
if (acceptedLanguages.includes(subdomain))
|
||||
// adding lang param
|
||||
link = `${rest}${rest.includes('?') ? '&' : '?'}lang=${subdomain}`;
|
||||
else if (subdomain === 'www') link = rest; // doing nothing
|
||||
else link = `/space/${subdomain}${rest}`; // gotta be a space url.
|
||||
|
||||
return res.redirect(link);
|
||||
};
|
||||
|
||||
export const unimplemented = (req, res, next) => {
|
||||
const data = {
|
||||
message: "This route isn't yet implemented. Check back sometime later!",
|
||||
statusCode: 501,
|
||||
};
|
||||
|
||||
res.status(data.statusCode).render('error', {
|
||||
data,
|
||||
meta: {
|
||||
title: 'Not yet implemented',
|
||||
url: `${req.urlObj.origin}${req.urlObj.pathname}`,
|
||||
url: req.urlObj,
|
||||
imageUrl: `${req.urlObj.origin}/icon.svg`,
|
||||
urlObj: req.urlObj,
|
||||
description: message,
|
||||
description: data.message,
|
||||
},
|
||||
});
|
||||
};
|
||||
|
|
|
@ -3,45 +3,55 @@
|
|||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
import * as cheerio from 'cheerio';
|
||||
import axiosInstance from '../utils/axiosInstance.js';
|
||||
import getAxiosInstance from '../utils/getAxiosInstance.js';
|
||||
import AppError from '../utils/AppError.js';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// FUNCTION
|
||||
////////////////////////////////////////////////////////
|
||||
/**
|
||||
*
|
||||
* makes a call to quora.com(with the resourceStr appended) and returns parsed JSON containing the data about the resource requested.
|
||||
* @param {string} resourceStr a string after the baseURL
|
||||
* @param {{keyword: string, lang?: string, toEncode?: boolean}} options additional options
|
||||
* @returns JSON containing the result
|
||||
* @description makes a call to quora.com(with the resourceStr appended) and returns parsed JSON containing the data about the resource requested.
|
||||
* @example await fetcher('What-is-free-and-open-software'); // will return object containing answers
|
||||
* await fetcher('topic/Space-Physics'); // will return 'space physics' topic object
|
||||
* await fetcher('profile/Charlie-Cheever'); // will return object containing information about charlie cheever
|
||||
*/
|
||||
const fetcher = async resourceStr => {
|
||||
const fetcher = async (
|
||||
resourceStr,
|
||||
{ keyword, lang, toEncode = true }
|
||||
) => {
|
||||
try {
|
||||
// as url might contain unescaped chars. so, encodeing it right away
|
||||
const res = await axiosInstance.get(encodeURIComponent(resourceStr));
|
||||
|
||||
// as url might contain unescaped chars. so, encoding it right away
|
||||
const str = toEncode ? encodeURIComponent(resourceStr) : resourceStr;
|
||||
const axiosInstance = getAxiosInstance(lang);
|
||||
const res = await axiosInstance.get(str);
|
||||
|
||||
const $ = cheerio.load(res.data);
|
||||
|
||||
// this logic is prone to breakage as Quora changes position of the script that includes answers.
|
||||
// Cur position: 4th from bottom.
|
||||
const rawData = $('body script:nth-last-child(4)')
|
||||
.html()
|
||||
?.match(/"\{.*\}"/m)?.[0];
|
||||
const regex = new RegExp(`"{\\\\"data\\\\":\\{\\\\"${keyword}.*\\}"`); // equivalent to /"\{\\"data\\":\{\\"searchConnection.*\}"/
|
||||
|
||||
if (!rawData || !Object.entries(rawData).length)
|
||||
throw new AppError("couldn't retrieve data", 500);
|
||||
let rawData;
|
||||
$('body script').each((i, el) => {
|
||||
const extractedVal = $(el).html().match(regex)?.[0];
|
||||
|
||||
const data = JSON.parse(rawData);
|
||||
if (extractedVal) {
|
||||
rawData = extractedVal;
|
||||
return false; // breaks loop
|
||||
}
|
||||
return true;
|
||||
});
|
||||
|
||||
return data;
|
||||
if (!rawData) throw new AppError("couldn't retrieve data", 500);
|
||||
|
||||
return JSON.parse(rawData);
|
||||
} catch (err) {
|
||||
if (err.response?.status === 404) throw new AppError('Not found', 404);
|
||||
else if (err.response?.status === 429)
|
||||
const statusCode = err.response?.status;
|
||||
if (statusCode === 404) throw new AppError('Not found', 404);
|
||||
else if (statusCode === 429 || statusCode === 403)
|
||||
throw new AppError(
|
||||
'Quora is rate limiting this instance. Consider hosting your own. Instructions are at Github',
|
||||
'Quora is rate limiting this instance. Try another or host your own.',
|
||||
503
|
||||
);
|
||||
else throw err;
|
||||
|
|
|
@ -3,17 +3,20 @@
|
|||
////////////////////////////////////////////////////////
|
||||
// import log from '../utils/log.js';
|
||||
import AppError from '../utils/AppError.js';
|
||||
import { quetrefy } from '../utils/urlModifiers.js';
|
||||
import fetcher from './fetcher.js';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// FUNCTION
|
||||
////////////////////////////////////////////////////////
|
||||
const getAnswers = async slug => {
|
||||
const KEYWORD = 'question';
|
||||
|
||||
const getAnswers = async (slug, lang) => {
|
||||
// getting data and destructuring it in case it exists
|
||||
const res = await fetcher(slug);
|
||||
const res = await fetcher(slug, { keyword: KEYWORD, lang });
|
||||
|
||||
const {
|
||||
data: { question: rawData },
|
||||
data: { [KEYWORD]: rawData },
|
||||
} = JSON.parse(res);
|
||||
|
||||
if (!rawData)
|
||||
|
@ -42,14 +45,14 @@ const getAnswers = async slug => {
|
|||
isAnon: ansObj.node.answer.author.isAnon,
|
||||
image: ansObj.node.answer.author.profileImageUrl,
|
||||
isVerified: ansObj.node.answer.author.isVerified,
|
||||
url: ansObj.node.answer.author.profileUrl,
|
||||
url: quetrefy(ansObj.node.answer.author.profileUrl),
|
||||
name: `${ansObj.node.answer.author.names[0].givenName} ${ansObj.node.answer.author.names[0].familyName}`,
|
||||
credential: ansObj.node.answer.authorCredential?.translatedString,
|
||||
// additionalCredentials: ansObj.node.answer?.credibilityFacts.map(),
|
||||
},
|
||||
originalQuestion: {
|
||||
text: JSON.parse(ansObj.node.answer.question.title).sections,
|
||||
url: ansObj.node.answer.question.url,
|
||||
url: quetrefy(ansObj.node.answer.question.url),
|
||||
qid: ansObj.node.answer.question.qid,
|
||||
isDeleted: ansObj.node.answer.question.isDeleted,
|
||||
},
|
||||
|
@ -59,7 +62,7 @@ const getAnswers = async slug => {
|
|||
const data = {
|
||||
question: {
|
||||
text: JSON.parse(rawData.title).sections,
|
||||
url: rawData.url,
|
||||
url: quetrefy(rawData.url),
|
||||
qid: rawData.qid,
|
||||
idDeleted: rawData.isDeleted,
|
||||
isViewable: rawData.isVisibleToViewer,
|
||||
|
@ -70,12 +73,12 @@ const getAnswers = async slug => {
|
|||
topics: rawData.topics.map(topicObj => ({
|
||||
tid: topicObj.tid,
|
||||
name: topicObj.name,
|
||||
url: topicObj.url,
|
||||
url: quetrefy(topicObj.url),
|
||||
})),
|
||||
relatedQuestions: rawData.bottomRelatedQuestionsInfo.relatedQuestions.map(
|
||||
questionObj => ({
|
||||
qid: questionObj.qid,
|
||||
url: questionObj.url,
|
||||
url: quetrefy(questionObj.url),
|
||||
text: JSON.parse(questionObj.title).sections,
|
||||
})
|
||||
),
|
||||
|
|
|
@ -2,6 +2,7 @@
|
|||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
import AppError from '../utils/AppError.js';
|
||||
import { quetrefy } from '../utils/urlModifiers.js';
|
||||
import fetcher from './fetcher.js';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
|
@ -28,14 +29,14 @@ const feedAnswerCleaner = answer => ({
|
|||
isAnon: answer.author.isAnon,
|
||||
image: answer.author.profileImageUrl,
|
||||
isVerified: answer.author.isVerified,
|
||||
url: answer.author.profileUrl,
|
||||
url: quetrefy(answer.author.profileUrl),
|
||||
name: `${answer.author.names[0].givenName} ${answer.author.names[0].familyName}`,
|
||||
credential: answer.authorCredential?.translatedString,
|
||||
// additionalCredentials: answer?.credibilityFacts.map(),
|
||||
},
|
||||
originalQuestion: {
|
||||
question: {
|
||||
text: JSON.parse(answer.question.title).sections,
|
||||
url: answer.question.url,
|
||||
url: quetrefy(answer.question.url),
|
||||
qid: answer.question.qid,
|
||||
isDeleted: answer.question.isDeleted,
|
||||
},
|
||||
|
@ -45,7 +46,7 @@ const feedPostCleaner = post => ({
|
|||
isPinned: post.isPinned,
|
||||
pid: post.pid,
|
||||
isViewable: post.viewerHasAccess,
|
||||
url: post.url,
|
||||
url: quetrefy(post.url),
|
||||
title: JSON.parse(post.title).sections,
|
||||
isDeleted: post.isDeleted,
|
||||
text: JSON.parse(post.content).sections,
|
||||
|
@ -61,14 +62,14 @@ const feedPostCleaner = post => ({
|
|||
image: post.author.profileImageUrl,
|
||||
isVerified: post.author.isVerified,
|
||||
isPlusUser: post.author.consumerBundleActive,
|
||||
url: post.author.profileUrl,
|
||||
url: quetrefy(post.author.profileUrl),
|
||||
name: `${post.author.names[0].givenName} ${post.author.names[0].familyName}`,
|
||||
credential: post.authorCredential?.translatedString,
|
||||
},
|
||||
...(post.tribeItem && {
|
||||
space: {
|
||||
name: post.tribeItem.tribe.nameString,
|
||||
url: post.tribeItem.tribe.url,
|
||||
url: quetrefy(post.tribeItem.tribe.url),
|
||||
image: post.tribeItem.tribe.iconRetinaUrl,
|
||||
description: post.tribeItem.descriptionString,
|
||||
numFollowers: post.tribeItem.tribe.numFollowers,
|
||||
|
@ -79,7 +80,7 @@ const feedQuestionCleaner = question => ({
|
|||
type: 'question',
|
||||
text: JSON.parse(question.title).sections,
|
||||
qid: question.qid,
|
||||
url: question.url,
|
||||
url: quetrefy(question.url),
|
||||
isDeleted: question.isDeleted,
|
||||
numFollowers: question.followerCount,
|
||||
creationTime: question.creationTime,
|
||||
|
@ -105,12 +106,14 @@ const feedCleaner = feed => {
|
|||
////////////////////////////////////////////////////////
|
||||
// FUNCTION
|
||||
////////////////////////////////////////////////////////
|
||||
const getProfile = async slug => {
|
||||
const KEYWORD = 'user';
|
||||
|
||||
const getProfile = async (slug, lang) => {
|
||||
// getting data and destructuring it in case it exists
|
||||
const res = await fetcher(`profile/${slug}`);
|
||||
const res = await fetcher(`profile/${slug}`, { keyword: KEYWORD, lang });
|
||||
|
||||
const {
|
||||
data: { user: rawData },
|
||||
data: { [KEYWORD]: rawData },
|
||||
} = JSON.parse(res);
|
||||
|
||||
if (!rawData)
|
||||
|
@ -125,7 +128,7 @@ const getProfile = async slug => {
|
|||
uid: rawData.uid,
|
||||
image: rawData.profileImageUrl,
|
||||
name: `${rawData.names[0].givenName} ${rawData.names[0].familyName}`,
|
||||
profile: rawData.profileUrl,
|
||||
profile: quetrefy(rawData.profileUrl),
|
||||
isDeceased: rawData.isDeceased,
|
||||
isBusiness: rawData.businessStatus,
|
||||
isBot: rawData.isUserBot,
|
||||
|
@ -184,7 +187,7 @@ const getProfile = async slug => {
|
|||
numFollowingSpaces: rawData.numFollowedTribes,
|
||||
spaces: rawData.followingTribesConnection.edges.map(space => ({
|
||||
numItems: space.node.numItemsOfUser,
|
||||
url: space.node.url,
|
||||
url: quetrefy(space.node.url),
|
||||
name: space.node.nameString,
|
||||
image: space.node.iconRetinaUrl,
|
||||
isSensitive: space.node.isSensitive,
|
||||
|
@ -194,7 +197,7 @@ const getProfile = async slug => {
|
|||
numFollowingTopics: rawData.numFollowedTopics,
|
||||
topics: rawData.expertiseTopicsConnection.edges.map(topic => ({
|
||||
name: topic.node.name,
|
||||
url: topic.node.url,
|
||||
url: quetrefy(topic.node.url),
|
||||
isSensitive: topic.node.isSensitive,
|
||||
numFollowers: topic.node.numFollowers,
|
||||
image: topic.node.photoUrl,
|
||||
|
|
169
fetchers/getSearch.js
Normal file
169
fetchers/getSearch.js
Normal file
|
@ -0,0 +1,169 @@
|
|||
////////////////////////////////////////////////////////
|
||||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
import AppError from '../utils/AppError.js';
|
||||
import fetcher from './fetcher.js';
|
||||
import { quetrefy } from '../utils/urlModifiers.js';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// HELPER FUNCTIONS
|
||||
////////////////////////////////////////////////////////
|
||||
const topicCleaner = topic => ({
|
||||
type: 'topic',
|
||||
url: quetrefy(topic.url),
|
||||
name: topic.name,
|
||||
numFollowers: topic.numFollowers,
|
||||
image: topic.photoUrl,
|
||||
isSensitive: topic.isSensitive,
|
||||
});
|
||||
const spaceCleaner = space => ({
|
||||
type: 'space',
|
||||
numUsers: space.tribeUserCount,
|
||||
url: quetrefy(space.url),
|
||||
name: space.nameString,
|
||||
description: space.descriptionString,
|
||||
image: space.iconRetinaUrl,
|
||||
isSensitive: space.isSensitive,
|
||||
});
|
||||
const profileCleaner = profile => ({
|
||||
type: 'profile',
|
||||
credential: profile.bestCredential?.translatedString,
|
||||
isAnon: profile.isAnon,
|
||||
name: `${profile.names[0]?.givenName} ${profile.names[0]?.familyName}`,
|
||||
url: quetrefy(profile.profileUrl),
|
||||
image: profile.profileImageUrl,
|
||||
numFollowers: profile.followerCount,
|
||||
isVerified: profile.isVerified,
|
||||
isBusiness: profile.businessStatus,
|
||||
isPlusUser: profile.consumerBundleActive,
|
||||
});
|
||||
const questionCleaner = question => ({
|
||||
type: 'question',
|
||||
text: JSON.parse(question.title).sections,
|
||||
url: quetrefy(question.url),
|
||||
isDeleted: question.isDeleted,
|
||||
numFollowers: question.followerCount,
|
||||
creationTime: question.creationTime,
|
||||
numComments: question.numDisplayComments,
|
||||
isSensitive: question.isSensitive,
|
||||
});
|
||||
const answerCleaner = ({ question, previewAnswer: answer }) => ({
|
||||
type: 'answer',
|
||||
question: {
|
||||
...questionCleaner(question),
|
||||
},
|
||||
...(answer.originalQuestionIfDifferent && {
|
||||
originalQuestion: {
|
||||
text: JSON.parse(answer.originalQuestionIfDifferent.question.title)
|
||||
.sections,
|
||||
url: quetrefy(answer.originalQuestionIfDifferent.question.url),
|
||||
qid: answer.originalQuestionIfDifferent.question.qid,
|
||||
},
|
||||
}),
|
||||
isViewable: !!answer.viewerHasAccess,
|
||||
text: JSON.parse(answer.content).sections,
|
||||
creationTime: answer.creationTime,
|
||||
updatedTime: answer.updatedTime,
|
||||
numComments: answer.numDisplayComments,
|
||||
numUpvotes: answer.numUpvotes,
|
||||
numViews: answer.numViews,
|
||||
numShares: answer.numSharers,
|
||||
numAnswerRequests: answer.numRequesters,
|
||||
isBusinessAnswer: answer.businessAnswer,
|
||||
url: quetrefy(answer.url),
|
||||
isSensitive: answer.isSensitive,
|
||||
author: {
|
||||
uid: answer.author.uid,
|
||||
isAnon: answer.author.isAnon,
|
||||
image: answer.author.profileImageUrl,
|
||||
isVerified: answer.author.isVerified,
|
||||
isPlusUser: answer.author.consumerBundleActive,
|
||||
url: quetrefy(answer.author.profileUrl),
|
||||
name: `${answer.author.names[0].givenName} ${answer.author.names[0].familyName}`,
|
||||
credential: answer.authorCredential?.translatedString,
|
||||
},
|
||||
});
|
||||
const postCleaner = post => ({
|
||||
type: 'post',
|
||||
pid: post.pid,
|
||||
isViewable: post.viewerHasAccess,
|
||||
url: quetrefy(post.url),
|
||||
title: JSON.parse(post.title).sections,
|
||||
isDeleted: post.isDeleted,
|
||||
isSensitive: post.isSensitive,
|
||||
text: JSON.parse(post.content).sections,
|
||||
creationTime: post.creationTime,
|
||||
updatedTime: post.updatedTime,
|
||||
numComments: post.numDisplayComments,
|
||||
numUpvotes: post.numUpvotes,
|
||||
numViews: post.numViews,
|
||||
numShares: post.numSharers,
|
||||
author: {
|
||||
uid: post.author.uid,
|
||||
isAnon: post.author.isAnon,
|
||||
image: post.author.profileImageUrl,
|
||||
isVerified: post.author.isVerified,
|
||||
isPlusUser: post.author.consumerBundleActive,
|
||||
url: quetrefy(post.author.profileUrl),
|
||||
name: `${post.author.names[0].givenName} ${post.author.names[0].familyName}`,
|
||||
credential: post.authorCredential?.translatedString,
|
||||
},
|
||||
...(post.tribeItem && {
|
||||
space: {
|
||||
isSensitive: post.tribeItem.tribe.isSensitive,
|
||||
name: post.tribeItem.tribe.nameString,
|
||||
url: quetrefy(post.tribeItem.tribe.url),
|
||||
image: post.tribeItem.tribe.iconRetinaUrl,
|
||||
description: post.tribeItem.descriptionString,
|
||||
numFollowers: post.tribeItem.tribe.numFollowers,
|
||||
},
|
||||
}),
|
||||
});
|
||||
|
||||
const resultsCleaner = results => {
|
||||
const cleanedResults = results.map(result => {
|
||||
const resultToClean = result.node;
|
||||
|
||||
if (resultToClean.topic) return topicCleaner(resultToClean.topic);
|
||||
if (resultToClean.tribe) return spaceCleaner(resultToClean.tribe);
|
||||
if (resultToClean.post) return postCleaner(resultToClean.post);
|
||||
if (resultToClean.user) return profileCleaner(resultToClean.user);
|
||||
if (resultToClean.previewAnswer) return answerCleaner(resultToClean);
|
||||
if (resultToClean.question) return questionCleaner(resultToClean.question);
|
||||
|
||||
return {};
|
||||
});
|
||||
|
||||
return cleanedResults;
|
||||
};
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// FUNCTION
|
||||
////////////////////////////////////////////////////////
|
||||
const KEYWORD = 'searchConnection';
|
||||
|
||||
const getSearch = async (querySlug, lang) => {
|
||||
const options = { keyword: KEYWORD, lang, toEncode: false };
|
||||
const res = await fetcher(`search/${querySlug}`, options);
|
||||
|
||||
const {
|
||||
data: { [KEYWORD]: rawData },
|
||||
} = JSON.parse(res);
|
||||
|
||||
if (!rawData)
|
||||
throw new AppError(
|
||||
"Search couldn't be done. Recheck the URL, or resend the request if you believe the URL is correct.",
|
||||
404
|
||||
);
|
||||
|
||||
const data = {
|
||||
results: resultsCleaner(rawData.edges),
|
||||
};
|
||||
|
||||
return data;
|
||||
};
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// EXPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
export default getSearch;
|
|
@ -2,17 +2,21 @@
|
|||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
import AppError from '../utils/AppError.js';
|
||||
import { quetrefy } from '../utils/urlModifiers.js';
|
||||
import fetcher from './fetcher.js';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// FUNCTION
|
||||
////////////////////////////////////////////////////////
|
||||
const getTopic = async slug => {
|
||||
|
||||
const KEYWORD = 'topic';
|
||||
|
||||
const getTopic = async (slug, lang) => {
|
||||
// getting data and destructuring it in case it exists, else throwing an error
|
||||
const res = await fetcher(`topic/${slug}`);
|
||||
const res = await fetcher(`topic/${slug}`, { keyword: KEYWORD, lang });
|
||||
|
||||
const {
|
||||
data: { topic: rawData },
|
||||
data: { [KEYWORD]: rawData },
|
||||
} = JSON.parse(res);
|
||||
|
||||
if (!rawData)
|
||||
|
@ -24,7 +28,7 @@ const getTopic = async slug => {
|
|||
const data = {
|
||||
tid: rawData.tid,
|
||||
name: rawData.name,
|
||||
url: rawData.url,
|
||||
url: quetrefy(rawData.url),
|
||||
image: rawData.photoUrl,
|
||||
aliases: rawData.aliases,
|
||||
numFollowers: rawData.numFollowers,
|
||||
|
@ -34,7 +38,7 @@ const getTopic = async slug => {
|
|||
mostViewedAuthors: rawData.mostViewedAuthors.map(author => ({
|
||||
uid: author.user.uid,
|
||||
name: `${author.user.names[0].givenName} ${author.user.names[0].familyName}`,
|
||||
profile: author.user.profileUrl,
|
||||
profile: quetrefy(author.user.profileUrl),
|
||||
image: author.user.profileImageUrl,
|
||||
isAnon: author.user.isAnon,
|
||||
isVerified: author.user.isVerified,
|
||||
|
@ -46,7 +50,7 @@ const getTopic = async slug => {
|
|||
relatedTopics: rawData.relatedTopics.map(topic => ({
|
||||
tid: topic.tid,
|
||||
name: topic.name,
|
||||
url: topic.url,
|
||||
url: quetrefy(topic.url),
|
||||
image: topic.photoUrl,
|
||||
numFollowers: topic.numFollowers,
|
||||
})),
|
||||
|
|
28
package.json
28
package.json
|
@ -1,6 +1,6 @@
|
|||
{
|
||||
"name": "quetre",
|
||||
"version": "4.0.0",
|
||||
"version": "5.5.0",
|
||||
"description": "a libre front-end for Quora",
|
||||
"private": true,
|
||||
"type": "module",
|
||||
|
@ -10,9 +10,9 @@
|
|||
"sass:build": "sass views/sass/main.scss:public/css/styles.css --style=compressed",
|
||||
"server:dev": "NODE_ENV=development nodemon server.js",
|
||||
"server:prod": "nodemon server.js",
|
||||
"dev": "(pnpm sass:watch) & (pnpm server:dev)",
|
||||
"prod": "(pnpm sass:build) & (pnpm server:prod)",
|
||||
"start": "(npm run sass:build) & (node server.js)"
|
||||
"dev": "pnpm sass:watch & pnpm server:dev",
|
||||
"prod": "pnpm sass:build && pnpm server:prod",
|
||||
"start": "pnpm run sass:build && node server.js"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
|
@ -31,22 +31,22 @@
|
|||
"axios": "^0.27.2",
|
||||
"cheerio": "1.0.0-rc.12",
|
||||
"compression": "^1.7.4",
|
||||
"dotenv": "^16.0.1",
|
||||
"express": "^4.18.1",
|
||||
"helmet": "^5.1.0",
|
||||
"dotenv": "^16.0.3",
|
||||
"express": "^4.18.2",
|
||||
"helmet": "^5.1.1",
|
||||
"morgan": "^1.10.0",
|
||||
"pug": "^3.0.2",
|
||||
"sass": "^1.54.0"
|
||||
"sass": "^1.57.1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@eslint/create-config": "^0.3.0",
|
||||
"@types/express": "^4.17.13",
|
||||
"eslint": "^8.20.0",
|
||||
"@eslint/create-config": "^0.3.1",
|
||||
"@types/express": "^4.17.15",
|
||||
"eslint": "^8.31.0",
|
||||
"eslint-config-airbnb-base": "^15.0.0",
|
||||
"eslint-config-prettier": "^8.5.0",
|
||||
"eslint-config-prettier": "^8.6.0",
|
||||
"eslint-plugin-import": "^2.26.0",
|
||||
"nodemon": "^2.0.19",
|
||||
"prettier": "^2.7.1"
|
||||
"nodemon": "^2.0.20",
|
||||
"prettier": "^2.8.2"
|
||||
},
|
||||
"nodemonConfig": {
|
||||
"ignore": [
|
||||
|
|
631
pnpm-lock.yaml
generated
631
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load diff
|
@ -48,3 +48,8 @@ btnTheme.addEventListener('click', () => {
|
|||
if (userPrefersTheme) setTheme(userPrefersTheme);
|
||||
else if (browserPrefersDarkTheme) setTheme('dark');
|
||||
})();
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// MATHJAX CONFIG
|
||||
////////////////////////////////////////////////////////
|
||||
window.MathJax = { options: { enableMenu: false } };
|
||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
public/mathjax/output/chtml/fonts/woff-v2/MathJax_Main-Bold.woff
Normal file
BIN
public/mathjax/output/chtml/fonts/woff-v2/MathJax_Main-Bold.woff
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
public/mathjax/output/chtml/fonts/woff-v2/MathJax_Zero.woff
Normal file
BIN
public/mathjax/output/chtml/fonts/woff-v2/MathJax_Zero.woff
Normal file
Binary file not shown.
1
public/mathjax/tex-chtml.js
Normal file
1
public/mathjax/tex-chtml.js
Normal file
File diff suppressed because one or more lines are too long
|
@ -1,4 +1,6 @@
|
|||
<svg width="0" height="0" class="hidden">
|
||||
<!-- TODO: readd links to fontawesome svg icons. They're ones that don't have viewBox of '0 0 24 24' -->
|
||||
|
||||
<!--main logo-->
|
||||
<symbol aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 189.91 130.7" id="icon-logo">
|
||||
<g transform="translate(-6.7989 -97.001)">
|
||||
|
@ -111,6 +113,12 @@
|
|||
<symbol aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" id="icon-quoraplus">
|
||||
<path d="M12,1L9,9L1,12L9,15L12,23L15,15L23,12L15,9L12,1Z"></path>
|
||||
</symbol>
|
||||
<symbol aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" id="icon-search">
|
||||
<path d="M9.5,3A6.5,6.5 0 0,1 16,9.5C16,11.11 15.41,12.59 14.44,13.73L14.71,14H15.5L20.5,19L19,20.5L14,15.5V14.71L13.73,14.44C12.59,15.41 11.11,16 9.5,16A6.5,6.5 0 0,1 3,9.5A6.5,6.5 0 0,1 9.5,3M9.5,5C7,5 5,7 5,9.5C5,12 7,14 9.5,14C12,14 14,12 14,9.5C14,7 12,5 9.5,5Z"></path>
|
||||
</symbol>
|
||||
<symbol aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" id="icon-cross">
|
||||
<path d="M13.46,12L19,17.54V19H17.54L12,13.46L6.46,19H5V17.54L10.54,12L5,6.46V5H6.46L12,10.54L17.54,5H19V6.46L13.46,12Z"></path>
|
||||
</symbol>
|
||||
|
||||
<!-- not yet used -->
|
||||
<symbol aria-hidden="true" focusable="false" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512" id="icon-cancel">
|
||||
|
|
Before Width: | Height: | Size: 30 KiB After Width: | Height: | Size: 31 KiB |
|
@ -6,16 +6,19 @@ import {
|
|||
topic,
|
||||
image,
|
||||
profile,
|
||||
search,
|
||||
} from '../controllers/apiController.js';
|
||||
|
||||
const apiRouter = express.Router();
|
||||
|
||||
apiRouter.get('/', about);
|
||||
apiRouter.get('/search', unimplemented);
|
||||
apiRouter.get('/(|search)', search);
|
||||
apiRouter.get('/about', about);
|
||||
apiRouter.get('/image/:domain/:path', image);
|
||||
apiRouter.get('/profile/:name', profile);
|
||||
apiRouter.get('/topic/:slug', topic);
|
||||
apiRouter.get('/unanswered/:slug', answers);
|
||||
apiRouter.get('/space/:name', unimplemented);
|
||||
apiRouter.get('/space/:name/:slug', unimplemented);
|
||||
apiRouter.get('/:slug', answers);
|
||||
|
||||
export default apiRouter;
|
||||
|
|
|
@ -6,16 +6,21 @@ import {
|
|||
topic,
|
||||
unimplemented,
|
||||
profile,
|
||||
search,
|
||||
redirect,
|
||||
} from '../controllers/viewController.js';
|
||||
|
||||
const viewRouter = express.Router();
|
||||
|
||||
viewRouter.get('/', about);
|
||||
viewRouter.get('/(|search)', search); // search on / or /search
|
||||
viewRouter.get('/about', about);
|
||||
viewRouter.get('/privacy', privacy);
|
||||
viewRouter.get('/search', unimplemented);
|
||||
viewRouter.get('/profile/:name', profile);
|
||||
viewRouter.get('/topic/:slug', topic);
|
||||
viewRouter.get('/unanswered/:slug', answers);
|
||||
viewRouter.get('/space/:name', unimplemented);
|
||||
viewRouter.get('/space/:name/:slug', unimplemented);
|
||||
viewRouter.get('/:slug', answers);
|
||||
viewRouter.get('/redirect/*', redirect); // eg: /redirect/https://www.quora.com/topic/linux
|
||||
|
||||
export default viewRouter;
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
////////////////////////////////////////////////////////
|
||||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
// eslint-disable-next-line no-unused-vars
|
||||
import dotenv from 'dotenv/config'; // importing .env vars
|
||||
import app from './app.js';
|
||||
import log from './utils/log.js';
|
||||
|
|
|
@ -1,28 +0,0 @@
|
|||
////////////////////////////////////////////////////////
|
||||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
import axios from 'axios';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// FUNCTION
|
||||
////////////////////////////////////////////////////////
|
||||
/**
|
||||
* @description an axios instance having base url already set
|
||||
*/
|
||||
const axiosInstance = axios.create({
|
||||
baseURL: 'https://www.quora.com',
|
||||
// conditionally adding headers to the request config using ES6 spreading and short-circuiting
|
||||
headers: {
|
||||
...(process.env.AXIOS_USER_AGENT && {
|
||||
'User-Agent': process.env.AXIOS_USER_AGENT,
|
||||
}),
|
||||
...(process.env.ACCEPT && {
|
||||
Accept: process.env.ACCEPT,
|
||||
}),
|
||||
},
|
||||
});
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// EXPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
export default axiosInstance;
|
|
@ -2,10 +2,45 @@
|
|||
// EXPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
|
||||
// some routes are accidentally thought of as slug for answered question. filtering those here.
|
||||
/**
|
||||
* some routes are accidentally thought of as slug for answered question. filtering those here.
|
||||
*/
|
||||
export const nonSlugRoutes = [
|
||||
'favicon.ico',
|
||||
'apple-touch-icon.png',
|
||||
'site.webmanifest',
|
||||
'icon.svg',
|
||||
];
|
||||
|
||||
/**
|
||||
* array of languages supported.
|
||||
*
|
||||
* see {@link https://help.quora.com/hc/en-us/articles/360015662751-What-languages-does-Quora-support- this help question} and {@link https://loc.gov/standards/iso639-2/ISO-639-2_utf-8.txt this list} for more.
|
||||
*
|
||||
*/
|
||||
export const acceptedLanguages = [
|
||||
'en', // English
|
||||
'es', // Spanish
|
||||
'fr', // French
|
||||
'de', // German
|
||||
'it', // Italian
|
||||
'ja', // Japanese
|
||||
'id', // Indonesian
|
||||
'pt', // Portuguese
|
||||
'hi', // Hindi
|
||||
'nl', // Dutch
|
||||
'da', // Danish
|
||||
'fi', // Finnish
|
||||
'nb', // Norwegian
|
||||
'sv', // Swedish
|
||||
'mr', // Marathi
|
||||
'bn', // Bengali
|
||||
'ta', // Tamil
|
||||
'ar', // Arabic
|
||||
'he', // Hebrew
|
||||
'gu', // Gujarati
|
||||
'kn', // Kannada
|
||||
'ml', // Malayalam
|
||||
'te', // Telugu
|
||||
'po', // Polish
|
||||
];
|
||||
|
|
31
utils/getAxiosInstance.js
Normal file
31
utils/getAxiosInstance.js
Normal file
|
@ -0,0 +1,31 @@
|
|||
////////////////////////////////////////////////////////
|
||||
// IMPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
import axios from 'axios';
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// FUNCTION
|
||||
////////////////////////////////////////////////////////
|
||||
/**
|
||||
* @description an axios instance having base url already set
|
||||
* @param {string} lang language to use. default is english.
|
||||
* @returns AxiosInstance
|
||||
*/
|
||||
const getAxiosInstance = (subdomain = 'www') =>
|
||||
axios.create({
|
||||
baseURL: `https://${subdomain}.quora.com`,
|
||||
// conditionally adding headers to the request config using ES6 spreading and short-circuiting
|
||||
headers: {
|
||||
...(process.env.AXIOS_USER_AGENT && {
|
||||
'User-Agent': process.env.AXIOS_USER_AGENT,
|
||||
}),
|
||||
...(process.env.ACCEPT && {
|
||||
Accept: process.env.ACCEPT,
|
||||
}),
|
||||
},
|
||||
});
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// EXPORTS
|
||||
////////////////////////////////////////////////////////
|
||||
export default getAxiosInstance;
|
|
@ -4,7 +4,7 @@
|
|||
|
||||
/**
|
||||
*
|
||||
* @param {string | {}} toLog stuff to log
|
||||
* @param {string | object} toLog stuff to log
|
||||
* @param {'success'| 'error'} type optional type param to color the log accordingly
|
||||
* @description logs color coded stuff to the stdout so that it's easily distinguishable
|
||||
*/
|
||||
|
@ -27,6 +27,7 @@ function log(toLog, type = null) {
|
|||
}
|
||||
|
||||
// actually logging to the console
|
||||
// eslint-disable-next-line no-console
|
||||
console.log(
|
||||
`\u001b[${data.colorCode}m ${data.emoji} ${data.message}\n${data.stack} \u001b[39m`
|
||||
);
|
||||
|
|
23
utils/urlModifiers.js
Normal file
23
utils/urlModifiers.js
Normal file
|
@ -0,0 +1,23 @@
|
|||
/* eslint-disable import/prefer-default-export */
|
||||
import { acceptedLanguages } from './constants.js';
|
||||
|
||||
/**
|
||||
* modifies link to be quetre-friendly.
|
||||
* @param {string} url the quora url to transform. could be relative or absolute
|
||||
*/
|
||||
export const quetrefy = url => {
|
||||
try {
|
||||
const link = new URL(url);
|
||||
const subdomain = link.hostname.split('.')[0];
|
||||
// normal url
|
||||
if (subdomain === 'www') return link.pathname;
|
||||
// lang specific route
|
||||
if (acceptedLanguages.includes(subdomain))
|
||||
return `${link.pathname}?lang=${subdomain}`;
|
||||
// must be spaces link
|
||||
return `/space/${subdomain}${link.pathname}`;
|
||||
} catch {
|
||||
// must be a relative url
|
||||
return url;
|
||||
}
|
||||
};
|
|
@ -38,19 +38,18 @@ html(lang='en')
|
|||
- let someAnswerContainsMath = false;
|
||||
|
||||
body.body
|
||||
-if (title !=='About')
|
||||
-if (meta.title !=='About')
|
||||
a.skip-link(href="#main") Skip to main content
|
||||
|
||||
//- MAIN CONTENT GOES HERE
|
||||
include layout/_header
|
||||
block content
|
||||
p placeholder text
|
||||
include layout/_footer
|
||||
|
||||
//- including mathjax script only when some answer has math expressions(using the var from above)
|
||||
if someAnswerContainsMath
|
||||
script#MathJax-script(
|
||||
type='text/javascript',
|
||||
src='https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-chtml.js',
|
||||
src='/mathjax/tex-chtml.js',
|
||||
async
|
||||
)
|
||||
)
|
||||
|
|
|
@ -3,19 +3,19 @@
|
|||
//-//////////////////////////////////////////////////////
|
||||
footer.footer(class=`${meta.title ==='About' ? 'footer__about' : ''}`)
|
||||
block footer
|
||||
|
||||
|
||||
//- EXTRA STUFF GOES HERE IF THE PAGE IS ABOUT PAGE
|
||||
|
||||
//- NAVIGATION
|
||||
nav.footer__nav-box(aria-label='Primary navigation')
|
||||
ul.footer__nav
|
||||
- if (meta.title !=='About')
|
||||
li.footer__nav-item: a.footer__nav-link.footer__link(href="/") About
|
||||
|
||||
li.footer__nav-item: a.footer__nav-link.footer__link(href="/about") About
|
||||
li.footer__nav-item: a.footer__nav-link.footer__link(href="/search") Search
|
||||
li.footer__nav-item: a.footer__nav-link.footer__link(href="https://github.com/zyachel/quetre") Source Code
|
||||
li.footer__nav-item: a.footer__nav-link.footer__link(href="/privacy") Privacy
|
||||
li.footer__nav-item: a.footer__nav-link.footer__link(href="#") Back to top
|
||||
|
||||
|
||||
//- LICENSE
|
||||
p.footer__license Licensed under
|
||||
a.footer__link(href="https://www.gnu.org/licenses/agpl-3.0.html") GNU AGPLv3
|
||||
|
|
|
@ -7,11 +7,14 @@ header.header(class=`${meta.title === 'About' ? 'header__about': ''}`)
|
|||
a.header__link.header__logo(href='/') Quetre
|
||||
//- for nav on about page
|
||||
block header__nav
|
||||
|
||||
|
||||
//- BUTTON FOR CHANGING THEME
|
||||
button.button.theme-changer.header__theme(aria-label='Change Theme')
|
||||
svg.icon.icon__theme.theme-changer__icon.theme-changer__icon--sun: use(href='/misc/sprite.svg#icon-sun')
|
||||
svg.icon.icon__theme.theme-changer__icon.theme-changer__icon--moon: use(href='/misc/sprite.svg#icon-moon')
|
||||
.header__misc
|
||||
a.link.header__search(href="/search", aria-label='search page')
|
||||
svg.icon: use(href='/misc/sprite.svg#icon-search')
|
||||
button.button.theme-changer.header__theme(aria-label='Change Theme')
|
||||
svg.icon.icon__theme.theme-changer__icon.theme-changer__icon--sun: use(href='/misc/sprite.svg#icon-sun')
|
||||
svg.icon.icon__theme.theme-changer__icon.theme-changer__icon--moon: use(href='/misc/sprite.svg#icon-moon')
|
||||
|
||||
//- IF THE PAGE IS ABOUT PAGE, THE BLOCK BELOW WILL GET POPULATED
|
||||
block header__info
|
|
@ -1,18 +1,18 @@
|
|||
//-//////////////////////////////////////////////////////
|
||||
//- INCLUDES/EXTENDS
|
||||
//-//////////////////////////////////////////////////////
|
||||
include ../mixins/_formatText
|
||||
include ../mixins/_metadata
|
||||
include ../mixins/_utils
|
||||
include _formatText
|
||||
include _metadata
|
||||
include _utils
|
||||
|
||||
//-//////////////////////////////////////////////////////
|
||||
//- MAIN CONTENT
|
||||
//-//////////////////////////////////////////////////////
|
||||
mixin addAnswer(answer)
|
||||
mixin addAnswer(answer, includeQuestion=false)
|
||||
article.answer
|
||||
//- ABOUT AUTHOR
|
||||
figure.metadata-primary
|
||||
figcaption.metadata-primary__heading
|
||||
.metadata-primary
|
||||
p.metadata-primary__heading
|
||||
if answer.author.isAnon
|
||||
span Anonymous
|
||||
else
|
||||
|
@ -21,14 +21,20 @@ mixin addAnswer(answer)
|
|||
svg.icon.metadata-primary__icon
|
||||
title verified
|
||||
use(href='/misc/sprite.svg#icon-verified')
|
||||
img.metadata-primary__image(src=answer.author.image.replace('https://', "/api/v1/image/"), alt=`${answer.author.name}'s profile photo`, loading='lazy')
|
||||
+proxifyImg(answer.author.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
p.metadata-primary__misc(aria-label=`${answer.author.name}'s credentials`)= answer.author.credential || ''
|
||||
|
||||
//- ORIGINAL QUESTION
|
||||
h3.answer__question.heading.heading__tertiary
|
||||
span Originally answered to
|
||||
a.answer__link.answers__link(href=answer.originalQuestion.url)
|
||||
+spansChecker(answer.originalQuestion.text[0].spans)
|
||||
if includeQuestion
|
||||
p.answer__question.heading.heading__tertiary
|
||||
span Answered to
|
||||
a.answer__link.answers__link(href=answer.question.url)
|
||||
+spansChecker(answer.question.text[0].spans)
|
||||
else if answer.originalQuestion
|
||||
p.answer__question.heading.heading__tertiary
|
||||
span Originally answered to
|
||||
a.answer__link.answers__link(href=answer.originalQuestion.url)
|
||||
+spansChecker(answer.originalQuestion.text[0].spans)
|
||||
|
||||
//- ANSWER
|
||||
section.answer__text.text__container
|
||||
|
|
|
@ -17,7 +17,7 @@ mixin spansChecker(spans)
|
|||
a.text__span-link.text__link(href=span.modifiers.embed.url)= span.modifiers.embed.title || '(link to user embedded content)'
|
||||
//- handle links
|
||||
- else if (span.modifiers.link) //- removing quora.com from the link in case it is a quora.com link.
|
||||
a.text__span-link.text__link(href=span.modifiers.link.url.split('https://www.quora.com')[1] || span.modifiers.link.url)=span.text
|
||||
+quetrefyUrl(span.modifiers.link.url, span.text)(class='text__span-link text__link')
|
||||
//- handle bold + italic text
|
||||
- else if (!!span.modifiers.bold && !!span.modifiers.italic)
|
||||
strong.text__span-bold: em.text__span-italic= span.text
|
||||
|
@ -42,7 +42,7 @@ mixin formatText(text)
|
|||
//- handle images
|
||||
-if(para.type ==='image')
|
||||
.text__para.text__image-container
|
||||
img.text__image(src=para.spans[0].modifiers.master_url.replace('https://', "/api/v1/image/"), alt='User embedded image', loading='lazy')
|
||||
+proxifyImg(para.spans[0].modifiers.master_url)(class='text__image', alt='User embedded image')
|
||||
//- handle code blocks
|
||||
- else if (para.type ==='code')
|
||||
pre.text__para.text__code: code
|
||||
|
@ -60,13 +60,13 @@ mixin formatText(text)
|
|||
hr.text__para.text__hr
|
||||
//- handling embedded youtube video
|
||||
- else if(para.type === 'yt-embed')
|
||||
.text__para.text__embed
|
||||
a.text__span-link.text__link(href=para.spans[0].modifiers.embed.url) (link to user-embedded YouTube video)
|
||||
.text__para.text__embed
|
||||
a.text__span-link.text__link(href=para.spans[0].modifiers.embed.url) (link to user-embedded YouTube video)
|
||||
//- handling embedded tweet
|
||||
- else if(para.type === 'tweet')
|
||||
.text__para.text__embed.text__embed-tweet
|
||||
//- directly pasting raw HTML which pug will process just fine
|
||||
| !{para.spans[0].modifiers.embed.html}
|
||||
.text__para.text__embed.text__embed-tweet
|
||||
//- directly pasting raw HTML which pug will process just fine
|
||||
| !{para.spans[0].modifiers.embed.html}
|
||||
//- rest of the types that don't need/can't be put in a semantic html tag
|
||||
- else
|
||||
p(class=`text__para text__${para.type}`)
|
||||
|
|
|
@ -1,8 +1,8 @@
|
|||
//-//////////////////////////////////////////////////////
|
||||
//- INCLUDES/EXTENDS
|
||||
//-//////////////////////////////////////////////////////
|
||||
include ../mixins/_formatText
|
||||
include ../mixins/_utils
|
||||
include _formatText
|
||||
include _utils
|
||||
|
||||
mixin addMetadataSecondary(iconName, name, number, numberType='number')
|
||||
p.metadata-secondary__item
|
||||
|
|
|
@ -1,9 +1,9 @@
|
|||
//-//////////////////////////////////////////////////////
|
||||
//- INCLUDES/EXTENDS
|
||||
//-//////////////////////////////////////////////////////
|
||||
include ../mixins/_formatText
|
||||
include ../mixins/_metadata
|
||||
include ../mixins/_utils
|
||||
include _formatText
|
||||
include _metadata
|
||||
include _utils
|
||||
|
||||
//-//////////////////////////////////////////////////////
|
||||
//- MAIN CONTENT
|
||||
|
@ -12,14 +12,14 @@ mixin addPost(post)
|
|||
article.answer
|
||||
//- ABOUT AUTHOR
|
||||
if post.space
|
||||
figure.answer__metdata-primary.metadata-primary
|
||||
figcaption.metadata-primary__heading
|
||||
.answer__metdata-primary.metadata-primary
|
||||
p.metadata-primary__heading
|
||||
a.answers__link(href=post.space.url)= post.space.name
|
||||
img.metadata-primary__image(src=post.space.image.replace('https://', "/api/v1/image/"), alt=`cover photo of ${post.space.name} space`, loading='lazy')
|
||||
+proxifyImg(post.space.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
p.metadata-primary__misc= post.space.description
|
||||
else
|
||||
figure.answer__metdata-primary.metadata-primary
|
||||
figcaption.metadata-primary__heading
|
||||
.answer__metdata-primary.metadata-primary
|
||||
p.metadata-primary__heading
|
||||
if post.author.isAnon
|
||||
span Anonymous
|
||||
else
|
||||
|
@ -28,7 +28,7 @@ mixin addPost(post)
|
|||
svg.metadata-primary__icon
|
||||
title verified
|
||||
use(href='/misc/sprite.svg#icon-verified')
|
||||
img.metadata-primary__image(src=post.author.image.replace('https://', "/api/v1/image/"), alt=`${post.author.name}'s profile photo`, loading='lazy')
|
||||
+proxifyImg(post.author.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
p.metadata-primary__misc(aria-label=`${post.author.name}'s credentials`)= post.author.credential || ''
|
||||
|
||||
//- POST HEADING
|
||||
|
|
|
@ -1,9 +1,9 @@
|
|||
//-//////////////////////////////////////////////////////
|
||||
//- INCLUDES/EXTENDS
|
||||
//-//////////////////////////////////////////////////////
|
||||
include ../mixins/_formatText
|
||||
include ../mixins/_metadata
|
||||
include ../mixins/_utils
|
||||
include _formatText
|
||||
include _metadata
|
||||
include _utils
|
||||
|
||||
//-//////////////////////////////////////////////////////
|
||||
//- MAIN CONTENT
|
||||
|
@ -19,5 +19,6 @@ mixin addQuestion(question)
|
|||
if question.lastFollowTime
|
||||
+addMetadataSecondary('clock-edit', 'Last followed', question.lastFollowTime, 'date')
|
||||
+addMetadataSecondary('users', 'Followers', question.numFollowers)
|
||||
+addMetadataSecondary('answers', 'Answers', question.numAnswers)
|
||||
if question.numAnswers
|
||||
+addMetadataSecondary('answers', 'Answers', question.numAnswers)
|
||||
+addMetadataSecondary('comments', 'Comments', question.numComments)
|
|
@ -8,3 +8,38 @@ mixin addDate(date, className='')
|
|||
|
||||
mixin formatNumber(number, className='')
|
||||
span(class=className)= new Intl.NumberFormat().format(number)
|
||||
|
||||
mixin proxifyImg(imgUrl)
|
||||
img(src=imgUrl.replace('https://', '/api/v1/image/'), loading='lazy')&attributes(attributes)
|
||||
|
||||
mixin quetrefyUrl(url, text)
|
||||
-
|
||||
const acceptedLanguages=['en','es','fr','de','it','ja','id','pt','hi','nl','da','fi','nb','sv','mr','bn','ta','ar','he','gu','kn','ml','te','po'];
|
||||
let link;
|
||||
const match = /^https:\/\/(.{2,})\.quora\.com(\/.*)$/.exec(url);
|
||||
|
||||
if (match) {
|
||||
const [_, subdomain, rest] = match;
|
||||
if (acceptedLanguages.includes(subdomain)) link = link = `${rest}${rest.includes('?') ? '&' : '?'}lang=${subdomain}`;
|
||||
else if (subdomain === 'www') link = rest;
|
||||
else link = `/space/${subdomain}${rest}`;
|
||||
} else {
|
||||
link = url;
|
||||
}
|
||||
|
||||
a(href=link)&attributes(attributes)= text
|
||||
|
||||
mixin quorafyUrl(url, text='View on Quora')
|
||||
-
|
||||
const link = new URL(`${url.pathname}${url.search}`, 'https://www.quora.com');
|
||||
const lang = link.searchParams.get('lang');
|
||||
const match = /(?<=^\/space\/)([^\/]+)\/(.*)$/.exec(link.pathname); // taking space name and rest of pathname out.
|
||||
if(lang) {
|
||||
link.hostname = `${lang}.quora.com`;
|
||||
link.searchParams.delete('lang');
|
||||
} else if(match) {
|
||||
link.hostname = `${match[1]}.quora.com`;
|
||||
link.pathname = match[2];
|
||||
}
|
||||
|
||||
a(href=link.href, rel='noreferrer', target='_blank')&attributes(attributes)= text
|
|
@ -86,12 +86,6 @@ block content
|
|||
summary.faqs__question There are some unreachable routes.
|
||||
svg.faqs__icon: use(href='/misc/sprite.svg#icon-open')
|
||||
p.faqs__answer I'm working to implement them soon.
|
||||
details.faqs__faq
|
||||
summary.faqs__question Why is website connecting to 'cdn.jsdelivr.net'?
|
||||
svg.faqs__icon: use(href='/misc/sprite.svg#icon-open')
|
||||
p.faqs__answer It is for an open source library –
|
||||
a.about__link(href='https://www.mathjax.org/') Mathjax
|
||||
| – which is used to display math eqations nicely. If I get enough time, I'll include it locally.
|
||||
details.faqs__faq
|
||||
summary.faqs__question Why are some math equations showing up weirdly?
|
||||
svg.faqs__icon: use(href='/misc/sprite.svg#icon-open')
|
||||
|
|
|
@ -4,6 +4,7 @@
|
|||
extends ../base
|
||||
include ../mixins/_formatText
|
||||
include ../mixins/_answer
|
||||
include ../mixins/_utils
|
||||
|
||||
//-//////////////////////////////////////////////////////
|
||||
//- MAIN CONTENT
|
||||
|
@ -18,7 +19,7 @@ block content
|
|||
.answers__metadata
|
||||
p.answers__answers-total= `${ data.numAnswers ? 'Total answers: ' + data.numAnswers : 'Unanswered'}`
|
||||
p.answers__answers-shown Viewable answers: #{data.answers.length}
|
||||
a.answers__question-link.answers__link(href='https://www.quora.com' + data.question.url) View on Quora
|
||||
+quorafyUrl(meta.url)(class='answers__question-link answers__link')
|
||||
|
||||
//- ANSWERS TO THIS QUESTION
|
||||
.answers-box.answers__answers-box
|
||||
|
|
|
@ -2,6 +2,7 @@
|
|||
//- INCLUDES/EXTENDS
|
||||
//-//////////////////////////////////////////////////////
|
||||
extends ../base
|
||||
include ../mixins/_utils
|
||||
|
||||
//-//////////////////////////////////////////////////////
|
||||
//- MAIN CONTENT
|
||||
|
@ -22,5 +23,5 @@ block content
|
|||
a.error__link(href="/") Home Page
|
||||
|.
|
||||
p.error__return Or view this route
|
||||
a.error__link(href="https://www.quora.com" + meta.urlObj.pathname) on Quora
|
||||
+quorafyUrl(meta.url, 'on Quora')(class='error__link')
|
||||
|.
|
|
@ -31,7 +31,3 @@ block content
|
|||
summary.faqs__question Data stored locally in your browser
|
||||
svg.faqs__icon: use(href='/misc/sprite.svg#icon-open')
|
||||
p.faqs__answer A key called 'theme' is stored in local storage provided by your browser to store your theme preference should you override the default theme. To prevent this behaviour, either disable JavaScript or local storage for Quetre.
|
||||
details.faqs__faq
|
||||
summary.faqs__question Data collected by other services
|
||||
svg.faqs__icon: use(href='/misc/sprite.svg#icon-open')
|
||||
p.faqs__answer If you're using the official instance (which is deployed on Heroku), Heroku might log your IP to prevent abuse. Also, as Quetre connects to 'cdn.jsdelivr.net' for the mathjax library, this service might log some data. So, follow due precaution. Using a VPN might be a good idea. Or even better, consider hosting your own instance.
|
||||
|
|
|
@ -36,7 +36,7 @@ block content
|
|||
section.profile__meta.profile-meta
|
||||
.profile-meta__basic
|
||||
.metadata-primary.profile-meta__about
|
||||
img.metadata-primary__image.profile-meta__image(src=data.basic.image.replace('https://', "/api/v1/image/"), alt=`${data.basic.name}'s profile photo`, loading='lazy')
|
||||
+proxifyImg(data.basic.image)(class='metadata-primary__image profile-meta__image', alt='', aria-hidden='true')
|
||||
h1.heading.heading__primary.metadata-primary__heading.profile-meta__heading-name.profile__name= data.basic.name
|
||||
if data.basic.isVerified
|
||||
svg.icon.metadata-primary__icon.profile-meta__icon
|
||||
|
@ -60,7 +60,7 @@ block content
|
|||
+addMetadataSecondary('flower', 'Deceased', '', 'empty')
|
||||
if data.basic.isBusiness
|
||||
+addMetadataSecondary('briefcase', 'Business', '', 'empty')
|
||||
a.link(href='https://www.quora.com' + data.basic.profile) View on Quora
|
||||
+quorafyUrl(meta.url)(class='link')
|
||||
|
||||
if data.profileFeed.description[0].spans[0].text
|
||||
.profile-meta__description
|
||||
|
@ -150,7 +150,7 @@ block content
|
|||
ul.profile-spaces__list
|
||||
each space in data.spaces.spaces
|
||||
li.metadata-primary.profile-spaces__list-item
|
||||
img.metadata-primary__image(src=space.image.replace('https://', '/api/v1/image/'), alt=`dedicated photo of ${space.name} space`)
|
||||
+proxifyImg(space.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
a.link.metadata-primary__heading(href=space.url)= space.name
|
||||
p.metadata-primary__misc
|
||||
+formatNumber(space.numItems)
|
||||
|
@ -166,7 +166,7 @@ block content
|
|||
ul.profile-topics__list
|
||||
each topic in data.topics.topics
|
||||
li.metadata-primary.profile-topics__list-item
|
||||
img.metadata-primary__image.profile-topics__img(src=topic.image.replace('https://', '/api/v1/image/'), alt=`image depicting ${topic.name} topic`)
|
||||
+proxifyImg(topic.image)(class='metadata-primary__image profile-topics__img', alt='', aria-hidden='true')
|
||||
a.link.metadata-primary__heading(href=topic.url)= topic.name
|
||||
p.metadata-primary__misc
|
||||
+formatNumber(topic.numAnswers)
|
||||
|
@ -185,7 +185,7 @@ block content
|
|||
//- svg.icon.profile__icon: use(href='/misc/sprite.svg#icon-verified')
|
||||
//- span pinned
|
||||
- if (item.type === 'answer')
|
||||
+addAnswer(item)
|
||||
+addAnswer(item, true)
|
||||
- else if (item.type === 'question')
|
||||
+addQuestion(item)
|
||||
- else if (item.type === 'post')
|
||||
|
|
89
views/pug/pages/search.pug
Normal file
89
views/pug/pages/search.pug
Normal file
|
@ -0,0 +1,89 @@
|
|||
//-//////////////////////////////////////////////////////
|
||||
//- INCLUDES/EXTENDS
|
||||
//-//////////////////////////////////////////////////////
|
||||
extends ../base
|
||||
include ../mixins/_formatText
|
||||
include ../mixins/_utils
|
||||
include ../mixins/_answer
|
||||
include ../mixins/_question
|
||||
include ../mixins/_post
|
||||
include ../mixins/_metadata
|
||||
|
||||
|
||||
//-//////////////////////////////////////////////////////
|
||||
//- MAIN CONTENT
|
||||
//-//////////////////////////////////////////////////////
|
||||
block content
|
||||
main#main(class=`main search ${data ? '' :'search--no-results'}`)
|
||||
-
|
||||
const typesArr = [{key: 'question', text: 'Questions'}, {key: 'answer', text: 'Answers'}, {key: 'post', text: 'Posts'}, {key: 'profile', text: 'Profiles'}, {key: 'topic', text: 'Topics'}, {key: 'tribe', text: 'Spaces'}]
|
||||
const timesArr = [{key: 'hour', text: 'Hour'}, {key: 'day', text: 'Day'}, {key: 'week', text: 'Week'}, {key: 'month', text: 'Month'}, {key: 'year', text: 'Year'}]
|
||||
const languagesArr = ['en','es','fr','de','it','ja','id','pt','hi','nl','da','fi','nb','sv','mr','bn','ta','ar','he','gu','kn','ml','te','po'];
|
||||
|
||||
form.search__form.search-form(action="/search", method='get', autocomplete='off', name='search')
|
||||
.search-form__search-container
|
||||
input.search-form__searchbar(type="search", name="q", placeholder='Enter your query...', minlength='3', aria-label='search for anything')
|
||||
button.search-form__button.search-form__button--reset(type="reset", aria-label='clear searchbar and filters'): svg.icon: use(href='/misc/sprite.svg#icon-cross')
|
||||
button.search-form__button.search-form__button--submit(type="submit", aria-label='search'): svg.icon: use(href='/misc/sprite.svg#icon-search')
|
||||
.search-form__filters-container.search-form__filters-container--type
|
||||
p.search-form__filters-heading Filter by Type
|
||||
each item in typesArr
|
||||
.search-form__filters-group
|
||||
input.search-form__radio.visually-hidden(type="radio", name="type", value=item.key, id=`type--${item.key}`)
|
||||
label.search-form__label(for=`type--${item.key}`)= item.text
|
||||
|
||||
.search-form__filters-container.search-form__filters-container--time
|
||||
p.search-form__filters-heading Filter by Time
|
||||
each item in timesArr
|
||||
.search-form__filters-group
|
||||
input.search-form__radio.visually-hidden(type="radio", name='time', value=item.key, id=`time--${item.key}`)
|
||||
label.search-form__label(for=`time--${item.key}`)= item.text
|
||||
|
||||
.search-form__filters-container.search-form__filters-container--lang
|
||||
p.search-form__filters-heading Filter by Language
|
||||
each lang in languagesArr
|
||||
.search-form__filters-group
|
||||
input.search-form__radio.visually-hidden(type="radio", name='lang', value=lang, id=`lang--${lang}`)
|
||||
label.search-form__label(for=`lang--${lang}`)= lang.toUpperCase()
|
||||
|
||||
//- TODO: refactor 'profile', 'topic', and 'space' into resusable mixins.
|
||||
- if (data?.results)
|
||||
section.search__results.search-results
|
||||
h1.heading.heading__primary Results
|
||||
- if (data.results.length === 0)
|
||||
p No results found for the query. Try being less specific and/or removing filters.
|
||||
- else
|
||||
.search-results__container
|
||||
each item in data.results
|
||||
.search-results__item
|
||||
- if (item.type === 'answer')
|
||||
+addAnswer(item, true)
|
||||
- else if (item.type === 'question')
|
||||
+addQuestion(item)
|
||||
- else if (item.type === 'post')
|
||||
+addPost(item)
|
||||
- else if (item.type === 'profile')
|
||||
.metadata-primary
|
||||
p.metadata-primary__heading
|
||||
if item.isAnon
|
||||
span Anonymous
|
||||
else
|
||||
a.link.metadata-primary__link(href=item.url)= item.name
|
||||
if item.isVerified
|
||||
svg.icon.metadata-primary__icon
|
||||
title verified
|
||||
use(href='/misc/sprite.svg#icon-verified')
|
||||
+proxifyImg(item.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
p.metadata-primary__misc(aria-label=`${item.name}'s credentials`)= item.credential || ''
|
||||
- else if (item.type === 'topic')
|
||||
.metadata-primary
|
||||
a.link.metadata-primary__heading(href=item.url)= item.name
|
||||
+proxifyImg(item.image)(class='metadata-primary__image', aria-hidden='true', alt='')
|
||||
p.metadata-primary__misc
|
||||
+formatNumber(item.numFollowers)
|
||||
| Followers
|
||||
- else if (item.type === 'space')
|
||||
.metadata-primary.profile-spaces__list-item
|
||||
+proxifyImg(item.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
a.link.metadata-primary__heading(href=item.url)= item.name
|
||||
p.metadata-primary__misc= item.description
|
|
@ -14,7 +14,7 @@ block content
|
|||
section.topic__stats.topic-stats
|
||||
.metadata-primary
|
||||
h1.heading.heading__primary.metadata-primary__heading.topic__name= data.name
|
||||
img.metadata-primary__image(src=data.image.replace('https://', "/api/v1/image/") alt=`cover photo of '${data.name}' topic`, loading='lazy')
|
||||
+proxifyImg(data.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
p.metadata-primary__misc
|
||||
if data.aliases.length
|
||||
span Also known as:
|
||||
|
@ -24,7 +24,7 @@ block content
|
|||
+addMetadataSecondary('question','Questions', data.numQuestions)
|
||||
if data.isAdult
|
||||
+addMetadataSecondary('danger', 'Adult Topic', '18+', true)
|
||||
a.link(href='https://www.quora.com' + data.url) View on Quora
|
||||
+quorafyUrl(meta.url)(class='link')
|
||||
|
||||
//- AUTHORS RELATED TO THE TOPIC AND METADATA
|
||||
.topic__famous-authors.famous-authors
|
||||
|
@ -32,8 +32,8 @@ block content
|
|||
.famous-authors__list
|
||||
each author in data.mostViewedAuthors
|
||||
.famous-authors__item
|
||||
figure.metadata-primary
|
||||
figcaption.metadata-primary__heading
|
||||
.metadata-primary
|
||||
p.metadata-primary__heading
|
||||
if author.isAnon
|
||||
span Anonymous
|
||||
else
|
||||
|
@ -42,7 +42,7 @@ block content
|
|||
svg.icon.metadata-primary__icon
|
||||
title verified
|
||||
use(href='/misc/sprite.svg#icon-verified')
|
||||
img.metadata-primary__image(src=author.image.replace('https://', "/api/v1/image/"), alt=`${author.name}'s profile photo`, loading='lazy')
|
||||
+proxifyImg(author.image)(class='metadata-primary__image', alt='', aria-hidden='true')
|
||||
p.metadata-primary__misc(aria-label=`${author.name}'s credentials`)= author.credential || ''
|
||||
.metadata-secondary
|
||||
+addMetadataSecondary('user', 'Followers', author.numFollowers)
|
||||
|
@ -56,7 +56,7 @@ block content
|
|||
each topic in data.relatedTopics
|
||||
.metadata-primary
|
||||
a.link.metadata-primary__heading(href=topic.url)= topic.name
|
||||
img.metadata-primary__image(src=topic.image.replace('https://', "/api/v1/image/") alt=`cover photo of '${topic.name}' topic`, loading='lazy')
|
||||
+proxifyImg(topic.image)(class='metadata-primary__image', aria-hidden='true', alt='')
|
||||
p.metadata-primary__misc
|
||||
+formatNumber(topic.numFollowers)
|
||||
| Followers
|
||||
|
|
|
@ -10,16 +10,22 @@
|
|||
$clr-primary: #e3f6f5;
|
||||
$clr-secondary: #272343;
|
||||
$clr-highlight: #ff5277;
|
||||
$ff-default: -apple-system, BlinkMacSystemFont, avenir next, avenir, segoe ui,
|
||||
helvetica neue, Cantarell, Ubuntu, roboto, noto, arial, sans-serif;
|
||||
$ff-monospace: Menlo, Consolas, Monaco, Liberation Mono, Lucida Console,
|
||||
monospace;
|
||||
|
||||
$misc-vars: (
|
||||
ff-primary: (
|
||||
'Worksans',
|
||||
sans-serif,
|
||||
$ff-default,
|
||||
),
|
||||
ff-alt-alpha: (
|
||||
'QuickSand',
|
||||
sans-serif,
|
||||
$ff-default,
|
||||
),
|
||||
ff-default: $ff-default,
|
||||
ff-mono: $ff-monospace,
|
||||
fs-160: 1.6rem,
|
||||
fs-180: 1.8rem,
|
||||
fs-200: 2rem,
|
||||
|
@ -45,6 +51,10 @@ $misc-vars: (
|
|||
);
|
||||
|
||||
$themed-vars: (
|
||||
color-scheme: (
|
||||
light: 'light',
|
||||
dark: 'dark',
|
||||
),
|
||||
// base
|
||||
clr-base-bg:
|
||||
(
|
||||
|
@ -152,5 +162,5 @@ $themed-vars: (
|
|||
clr-focus: (
|
||||
light: color.scale($clr-highlight, $lightness: 0%, $saturation: 0%),
|
||||
dark: color.scale($clr-highlight),
|
||||
)
|
||||
),
|
||||
);
|
||||
|
|
|
@ -43,6 +43,8 @@ body {
|
|||
// BASE STYLING
|
||||
////////////////////////////////////////////////////////
|
||||
:root {
|
||||
color-scheme: light dark;
|
||||
|
||||
// normal vars
|
||||
@include get-misc-vars;
|
||||
// themed vars(default:light)
|
||||
|
@ -61,9 +63,16 @@ body {
|
|||
@include get-themed-vars(dark);
|
||||
}
|
||||
}
|
||||
|
||||
// disabling fonts for webkit
|
||||
@include fix-for-safari {
|
||||
--ff-primary: var(--ff-default);
|
||||
--ff-alt-alpha: var(--ff-default);
|
||||
}
|
||||
}
|
||||
|
||||
.body {
|
||||
color-scheme: var(--color-scheme);
|
||||
font-size: var(--fs-160);
|
||||
font-family: var(--ff-primary);
|
||||
background-color: var(--clr-base-bg);
|
||||
|
@ -126,8 +135,7 @@ body {
|
|||
// KEYBOARD NAVIGATION
|
||||
////////////////////////////////////////////////////////
|
||||
:focus {
|
||||
outline: 3px solid var(--clr-focus);
|
||||
outline-offset: 0.2em;
|
||||
@include focus-rules;
|
||||
}
|
||||
|
||||
@supports selector(:focus-visible) {
|
||||
|
@ -136,30 +144,6 @@ body {
|
|||
}
|
||||
|
||||
:focus-visible {
|
||||
outline: 3px solid var(--clr-focus);
|
||||
outline-offset: 0.2em;
|
||||
@include focus-rules;
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// FOR MATHJAX
|
||||
////////////////////////////////////////////////////////
|
||||
mjx-container.MathJax {
|
||||
pointer-events: none;
|
||||
}
|
||||
/*
|
||||
.CtxtMenu_ContextMenu,
|
||||
.CtxtMenu_Info {
|
||||
background-color: var(--clr-base-bg) !important;
|
||||
font-family: inherit !important;
|
||||
color: inherit !important;
|
||||
}
|
||||
.CtxtMenu_InfoTitle,
|
||||
.CtxtMenu_InfoSignature,
|
||||
.CtxtMenu_MenuClose .CtxtMenu_InfoClose,
|
||||
.CtxtMenu_InfoContent {
|
||||
font-family: inherit !important;
|
||||
background: inherit !important;
|
||||
color: inherit !important;
|
||||
}
|
||||
*/
|
||||
|
|
|
@ -51,6 +51,8 @@
|
|||
.icon {
|
||||
max-height: var(--fs-500);
|
||||
max-width: var(--fs-500);
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
fill: var(--clr-base-icon);
|
||||
|
||||
&__down {
|
||||
|
@ -81,6 +83,23 @@
|
|||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// HELPER CLASSES
|
||||
////////////////////////////////////////////////////////
|
||||
.visually-hidden {
|
||||
clip: rect(1px, 1px, 1px, 1px) !important;
|
||||
border: 0 !important;
|
||||
-webkit-clip-path: inset(50%) !important;
|
||||
clip-path: inset(50%) !important;
|
||||
height: 1px !important;
|
||||
margin: -1px !important;
|
||||
overflow: hidden !important;
|
||||
padding: 0 !important;
|
||||
position: absolute !important;
|
||||
white-space: nowrap !important;
|
||||
width: 1px !important;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// LINKS
|
||||
////////////////////////////////////////////////////////
|
||||
|
@ -283,6 +302,11 @@
|
|||
list-style-type: none;
|
||||
|
||||
cursor: pointer;
|
||||
|
||||
// for hinding triangle on webkit
|
||||
&::-webkit-details-marker {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
&__icon {
|
||||
height: 1em;
|
||||
|
@ -507,3 +531,119 @@
|
|||
gap: var(--space-800);
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// SEARCH PAGE COMPONENTS
|
||||
////////////////////////////////////////////////////////
|
||||
///
|
||||
.search-form {
|
||||
display: grid;
|
||||
gap: var(--space-200);
|
||||
// justify-items: center;
|
||||
|
||||
&__search-container {
|
||||
display: grid;
|
||||
grid-auto-flow: column;
|
||||
grid-auto-columns: minmax(15rem, auto) 3rem 3rem;
|
||||
grid-auto-rows: 3rem;
|
||||
border: 1px solid var(--clr-base-icon-alt-alpha);
|
||||
border-radius: 100vmax;
|
||||
overflow: hidden;
|
||||
padding: var(--space-100);
|
||||
gap: var(--space-050);
|
||||
|
||||
&:focus-within {
|
||||
background-color: var(--clr-code-bg);
|
||||
color: var(--clr-code-text);
|
||||
}
|
||||
}
|
||||
|
||||
&__searchbar {
|
||||
outline: none;
|
||||
border: none;
|
||||
background-color: transparent;
|
||||
font: inherit;
|
||||
caret-color: var(--clr-base-icon);
|
||||
|
||||
// fix for browsers with non-standard properties. yes, webkit and blink suck.
|
||||
-webkit-appearance: none;
|
||||
&::-webkit-search-cancel-button {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
|
||||
&__button {
|
||||
border: none;
|
||||
background: transparent;
|
||||
cursor: pointer;
|
||||
border-radius: 100vmax;
|
||||
|
||||
&:hover {
|
||||
transform: scale(1.1);
|
||||
}
|
||||
&--reset {
|
||||
}
|
||||
|
||||
&--submit {
|
||||
}
|
||||
}
|
||||
|
||||
&__filters-container {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 2rem 1rem;
|
||||
}
|
||||
|
||||
&__filters-heading {
|
||||
flex-basis: 100%;
|
||||
}
|
||||
|
||||
&__filters-group {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
&__radio {
|
||||
}
|
||||
|
||||
&__label {
|
||||
cursor: pointer;
|
||||
background: var(--clr-code-bg);
|
||||
color: var(--clr-code-text);
|
||||
padding: 1rem;
|
||||
border-radius: 1rem;
|
||||
}
|
||||
|
||||
&__radio:checked + &__label {
|
||||
color: var(--clr-selection-text);
|
||||
background: var(--clr-selection-bg);
|
||||
}
|
||||
|
||||
&__radio:focus + &__label {
|
||||
@include focus-rules;
|
||||
}
|
||||
|
||||
@supports selector(:focus-visible) {
|
||||
&__radio:focus + &__label {
|
||||
outline: none;
|
||||
}
|
||||
|
||||
&__radio:focus-visible + &__label {
|
||||
@include focus-rules;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.search-results {
|
||||
--img-dim: var(--fs-600);
|
||||
|
||||
display: grid;
|
||||
gap: var(--space-500);
|
||||
|
||||
&__container {
|
||||
display: grid;
|
||||
gap: var(--space-500);
|
||||
}
|
||||
|
||||
&__item {
|
||||
}
|
||||
}
|
||||
|
|
|
@ -76,7 +76,13 @@
|
|||
}
|
||||
}
|
||||
|
||||
&__theme {
|
||||
&__misc {
|
||||
display: flex;
|
||||
gap: var(--space-200);
|
||||
}
|
||||
|
||||
&__theme,
|
||||
&__search {
|
||||
height: var(--fs-300);
|
||||
width: var(--fs-300);
|
||||
|
||||
|
@ -86,6 +92,12 @@
|
|||
}
|
||||
}
|
||||
|
||||
// search icon viewBox is larger than content, hence the fix
|
||||
&__search > svg {
|
||||
height: 105%;
|
||||
width: 105%;
|
||||
}
|
||||
|
||||
&__info {
|
||||
display: grid;
|
||||
place-items: center;
|
||||
|
|
|
@ -63,6 +63,11 @@
|
|||
}
|
||||
}
|
||||
|
||||
@mixin focus-rules {
|
||||
outline: 3px solid var(--clr-focus);
|
||||
outline-offset: 0.2em;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// BREAKPOINT MANAGER
|
||||
////////////////////////////////////////////////////////
|
||||
|
@ -104,3 +109,13 @@ $breakpoints: (
|
|||
@content;
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////
|
||||
// CHECK IF BROWSER IS SAFARI(it's the new IE)
|
||||
////////////////////////////////////////////////////////////////
|
||||
|
||||
@mixin fix-for-safari {
|
||||
@supports (-webkit-appearance: none) and (stroke-color: transparent) {
|
||||
@content;
|
||||
}
|
||||
}
|
|
@ -405,3 +405,54 @@
|
|||
padding-inline: var(--space-200);
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////
|
||||
// SEARCH
|
||||
////////////////////////////////////////////////////////
|
||||
.search {
|
||||
// justify-self: center;
|
||||
padding: var(--space-500) var(--space-800);
|
||||
|
||||
display: grid;
|
||||
grid-template-columns: 2fr 1.2fr;
|
||||
gap: var(--space-500);
|
||||
align-items: start;
|
||||
|
||||
&--no-results {
|
||||
grid-template-columns: unset;
|
||||
place-content: center;
|
||||
max-width: 100rem;
|
||||
margin-inline: auto;
|
||||
}
|
||||
|
||||
&__results {
|
||||
grid-column: 1 / 2;
|
||||
grid-row: 1 / 2;
|
||||
}
|
||||
|
||||
&__form {
|
||||
grid-row: 1 / 2;
|
||||
|
||||
grid-column: -2 / -1;
|
||||
}
|
||||
|
||||
@include respond-to(bp-1200) {
|
||||
padding: var(--space-500);
|
||||
gap: var(--space-500);
|
||||
}
|
||||
@include respond-to(bp-900) {
|
||||
grid-template-columns: auto;
|
||||
grid-template-rows: max-content auto;
|
||||
|
||||
&__results {
|
||||
grid-row: 2 / 3;
|
||||
}
|
||||
|
||||
&__form {
|
||||
grid-row: 1 / 2;
|
||||
}
|
||||
}
|
||||
@include respond-to(bp-550) {
|
||||
padding-inline: var(--space-200);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue