diff options
| author | 2022-02-02 00:15:44 +0100 | |
|---|---|---|
| committer | 2022-02-02 00:15:44 +0100 | |
| commit | 09fb2a4df59be324fc41176227ae3c684e2f6add (patch) | |
| tree | 9d5baa069f91419e3189115d143129b6dc4b8e31 | |
| parent | Merge pull request #649 from python-discord/update-pyfakefs (diff) | |
| parent | Duck pond removed when removing all filters. (diff) | |
Merge pull request #582 from python-discord/swfarnsworth/smarter-resources/merge-with-main
Smarter Resources
129 files changed, 2135 insertions, 825 deletions
| diff --git a/poetry.lock b/poetry.lock index 96e44c3b..3b26c275 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,47 +1,35 @@  [[package]]  name = "asgiref" -version = "3.4.1" +version = "3.5.0"  description = "ASGI specs, helper code, and adapters"  category = "main"  optional = false -python-versions = ">=3.6" +python-versions = ">=3.7"  [package.extras]  tests = ["pytest", "pytest-asyncio", "mypy (>=0.800)"]  [[package]]  name = "attrs" -version = "21.2.0" +version = "21.4.0"  description = "Classes Without Boilerplate"  category = "dev"  optional = false  python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"  [package.extras] -dev = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface", "furo", "sphinx", "sphinx-notfound-page", "pre-commit"] +dev = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface", "furo", "sphinx", "sphinx-notfound-page", "pre-commit", "cloudpickle"]  docs = ["furo", "sphinx", "zope.interface", "sphinx-notfound-page"] -tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface"] -tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins"] - -[[package]] -name = "backports.entry-points-selectable" -version = "1.1.1" -description = "Compatibility shim providing selectable entry points for older implementations" -category = "dev" -optional = false -python-versions = ">=2.7" - -[package.extras] -docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"] -testing = ["pytest", "pytest-flake8", "pytest-cov", "pytest-black (>=0.3.7)", "pytest-mypy", "pytest-checkdocs (>=2.4)", "pytest-enabler (>=1.0.1)"] +tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface", "cloudpickle"] +tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "cloudpickle"]  [[package]]  name = "bandit" -version = "1.7.1" +version = "1.7.2"  description = "Security oriented static analyser for python code."  category = "dev"  optional = false -python-versions = ">=3.5" +python-versions = ">=3.7"  [package.dependencies]  colorama = {version = ">=0.3.9", markers = "platform_system == \"Windows\""} @@ -49,6 +37,11 @@ GitPython = ">=1.0.1"  PyYAML = ">=5.3.1"  stevedore = ">=1.20.0" +[package.extras] +test = ["beautifulsoup4 (>=4.8.0)", "coverage (>=4.5.4)", "fixtures (>=3.0.0)", "flake8 (>=4.0.0)", "pylint (==1.9.4)", "stestr (>=2.5.0)", "testscenarios (>=0.5.0)", "testtools (>=2.3.0)", "toml"] +toml = ["toml"] +yaml = ["pyyaml"] +  [[package]]  name = "certifi"  version = "2021.10.8" @@ -67,7 +60,7 @@ python-versions = ">=3.6.1"  [[package]]  name = "charset-normalizer" -version = "2.0.9" +version = "2.0.11"  description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet."  category = "main"  optional = false @@ -138,7 +131,7 @@ bcrypt = ["bcrypt"]  [[package]]  name = "django-distill" -version = "2.9.1" +version = "2.9.2"  description = "Static site renderer and publisher for Django."  category = "main"  optional = false @@ -169,7 +162,7 @@ Django = ">=2.2"  [[package]]  name = "django-prometheus" -version = "2.1.0" +version = "2.2.0"  description = "Django middlewares to monitor your application with Prometheus.io."  category = "main"  optional = false @@ -214,11 +207,11 @@ python-versions = "*"  [[package]]  name = "filelock" -version = "3.4.0" +version = "3.4.2"  description = "A platform independent file lock."  category = "dev"  optional = false -python-versions = ">=3.6" +python-versions = ">=3.7"  [package.extras]  docs = ["furo (>=2021.8.17b43)", "sphinx (>=4.1)", "sphinx-autodoc-typehints (>=1.12)"] @@ -324,14 +317,14 @@ flake8 = "*"  [[package]]  name = "flake8-tidy-imports" -version = "4.5.0" +version = "4.6.0"  description = "A flake8 plugin that helps you write tidier imports."  category = "dev"  optional = false -python-versions = ">=3.6" +python-versions = ">=3.7"  [package.dependencies] -flake8 = ">=3.8.0,<5" +flake8 = ">=3.8.0"  [[package]]  name = "flake8-todo" @@ -357,7 +350,7 @@ smmap = ">=3.0.1,<6"  [[package]]  name = "gitpython" -version = "3.1.24" +version = "3.1.26"  description = "GitPython is a python library used to interact with Git repositories"  category = "dev"  optional = false @@ -365,7 +358,6 @@ python-versions = ">=3.7"  [package.dependencies]  gitdb = ">=4.0.1,<5" -typing-extensions = {version = ">=3.7.4.3", markers = "python_version < \"3.10\""}  [[package]]  name = "gunicorn" @@ -383,11 +375,11 @@ tornado = ["tornado (>=0.2)"]  [[package]]  name = "identify" -version = "2.4.0" +version = "2.4.6"  description = "File identification library for Python"  category = "dev"  optional = false -python-versions = ">=3.6.1" +python-versions = ">=3.7"  [package.extras]  license = ["ukkonen"] @@ -402,11 +394,11 @@ python-versions = ">=3.5"  [[package]]  name = "importlib-metadata" -version = "4.8.2" +version = "4.10.1"  description = "Read metadata from Python packages"  category = "main"  optional = false -python-versions = ">=3.6" +python-versions = ">=3.7"  [package.dependencies]  zipp = ">=0.5" @@ -414,7 +406,7 @@ zipp = ">=0.5"  [package.extras]  docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"]  perf = ["ipython"] -testing = ["pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "packaging", "pep517", "pyfakefs", "flufl.flake8", "pytest-perf (>=0.9.2)", "pytest-black (>=0.3.7)", "pytest-mypy", "importlib-resources (>=1.3)"] +testing = ["pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "packaging", "pyfakefs", "flufl.flake8", "pytest-perf (>=0.9.2)", "pytest-black (>=0.3.7)", "pytest-mypy", "importlib-resources (>=1.3)"]  [[package]]  name = "libsass" @@ -487,11 +479,11 @@ flake8-polyfill = ">=1.0.2,<2"  [[package]]  name = "platformdirs" -version = "2.4.0" +version = "2.4.1"  description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"."  category = "dev"  optional = false -python-versions = ">=3.6" +python-versions = ">=3.7"  [package.extras]  docs = ["Sphinx (>=4)", "furo (>=2021.7.5b38)", "proselint (>=0.10.2)", "sphinx-autodoc-typehints (>=1.12)"] @@ -499,7 +491,7 @@ test = ["appdirs (==1.4.4)", "pytest (>=6)", "pytest-cov (>=2.7)", "pytest-mock  [[package]]  name = "pre-commit" -version = "2.16.0" +version = "2.17.0"  description = "A framework for managing and maintaining multi-language pre-commit hooks."  category = "dev"  optional = false @@ -515,18 +507,18 @@ virtualenv = ">=20.0.8"  [[package]]  name = "prometheus-client" -version = "0.12.0" +version = "0.13.1"  description = "Python client for the Prometheus monitoring system."  category = "main"  optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +python-versions = ">=3.6"  [package.extras]  twisted = ["twisted"]  [[package]]  name = "psutil" -version = "5.8.0" +version = "5.9.0"  description = "Cross-platform lib for process and system monitoring in Python."  category = "dev"  optional = false @@ -625,7 +617,7 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"  [[package]]  name = "requests" -version = "2.26.0" +version = "2.27.1"  description = "Python HTTP for Humans."  category = "main"  optional = false @@ -734,16 +726,8 @@ optional = false  python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*"  [[package]] -name = "typing-extensions" -version = "4.0.1" -description = "Backported and Experimental Type Hints for Python 3.6+" -category = "dev" -optional = false -python-versions = ">=3.6" - -[[package]]  name = "urllib3" -version = "1.26.7" +version = "1.26.8"  description = "HTTP library with thread-safe connection pooling, file post, and more."  category = "main"  optional = false @@ -756,14 +740,13 @@ socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"]  [[package]]  name = "virtualenv" -version = "20.10.0" +version = "20.13.0"  description = "Virtual Python Environment builder"  category = "dev"  optional = false  python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"  [package.dependencies] -"backports.entry-points-selectable" = ">=1.0.4"  distlib = ">=0.3.1,<1"  filelock = ">=3.2,<4"  platformdirs = ">=2,<3" @@ -786,37 +769,33 @@ brotli = ["brotli"]  [[package]]  name = "zipp" -version = "3.6.0" +version = "3.7.0"  description = "Backport of pathlib-compatible object wrapper for zip files"  category = "main"  optional = false -python-versions = ">=3.6" +python-versions = ">=3.7"  [package.extras]  docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"] -testing = ["pytest (>=4.6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"] +testing = ["pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"]  [metadata]  lock-version = "1.1"  python-versions = "3.9.*" -content-hash = "61fa9b7aa99c2f03f4fe990633498bd6bba1f97e39b226f8005fd2a259faf852" +content-hash = "fc9b20c33c65a289122d710844285ac20d7598e65c7f8237f8903509f5b2dea4"  [metadata.files]  asgiref = [ -    {file = "asgiref-3.4.1-py3-none-any.whl", hash = "sha256:ffc141aa908e6f175673e7b1b3b7af4fdb0ecb738fc5c8b88f69f055c2415214"}, -    {file = "asgiref-3.4.1.tar.gz", hash = "sha256:4ef1ab46b484e3c706329cedeff284a5d40824200638503f5768edb6de7d58e9"}, +    {file = "asgiref-3.5.0-py3-none-any.whl", hash = "sha256:88d59c13d634dcffe0510be048210188edd79aeccb6a6c9028cdad6f31d730a9"}, +    {file = "asgiref-3.5.0.tar.gz", hash = "sha256:2f8abc20f7248433085eda803936d98992f1343ddb022065779f37c5da0181d0"},  ]  attrs = [ -    {file = "attrs-21.2.0-py2.py3-none-any.whl", hash = "sha256:149e90d6d8ac20db7a955ad60cf0e6881a3f20d37096140088356da6c716b0b1"}, -    {file = "attrs-21.2.0.tar.gz", hash = "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb"}, -] -"backports.entry-points-selectable" = [ -    {file = "backports.entry_points_selectable-1.1.1-py2.py3-none-any.whl", hash = "sha256:7fceed9532a7aa2bd888654a7314f864a3c16a4e710b34a58cfc0f08114c663b"}, -    {file = "backports.entry_points_selectable-1.1.1.tar.gz", hash = "sha256:914b21a479fde881635f7af5adc7f6e38d6b274be32269070c53b698c60d5386"}, +    {file = "attrs-21.4.0-py2.py3-none-any.whl", hash = "sha256:2d27e3784d7a565d36ab851fe94887c5eccd6a463168875832a1be79c82828b4"}, +    {file = "attrs-21.4.0.tar.gz", hash = "sha256:626ba8234211db98e869df76230a137c4c40a12d72445c45d5f5b716f076e2fd"},  ]  bandit = [ -    {file = "bandit-1.7.1-py3-none-any.whl", hash = "sha256:f5acd838e59c038a159b5c621cf0f8270b279e884eadd7b782d7491c02add0d4"}, -    {file = "bandit-1.7.1.tar.gz", hash = "sha256:a81b00b5436e6880fa8ad6799bc830e02032047713cbb143a12939ac67eb756c"}, +    {file = "bandit-1.7.2-py3-none-any.whl", hash = "sha256:e20402cadfd126d85b68ed4c8862959663c8c372dbbb1fca8f8e2c9f55a067ec"}, +    {file = "bandit-1.7.2.tar.gz", hash = "sha256:6d11adea0214a43813887bfe71a377b5a9955e4c826c8ffd341b494e3ab25260"},  ]  certifi = [      {file = "certifi-2021.10.8-py2.py3-none-any.whl", hash = "sha256:d62a0163eb4c2344ac042ab2bdf75399a71a2d8c7d47eac2e2ee91b9d6339569"}, @@ -827,8 +806,8 @@ cfgv = [      {file = "cfgv-3.3.1.tar.gz", hash = "sha256:f5a830efb9ce7a445376bb66ec94c638a9787422f96264c98edc6bdeed8ab736"},  ]  charset-normalizer = [ -    {file = "charset-normalizer-2.0.9.tar.gz", hash = "sha256:b0b883e8e874edfdece9c28f314e3dd5badf067342e42fb162203335ae61aa2c"}, -    {file = "charset_normalizer-2.0.9-py3-none-any.whl", hash = "sha256:1eecaa09422db5be9e29d7fc65664e6c33bd06f9ced7838578ba40d58bdf3721"}, +    {file = "charset-normalizer-2.0.11.tar.gz", hash = "sha256:98398a9d69ee80548c762ba991a4728bfc3836768ed226b3945908d1a688371c"}, +    {file = "charset_normalizer-2.0.11-py3-none-any.whl", hash = "sha256:2842d8f5e82a1f6aa437380934d5e1cd4fcf2003b06fed6940769c164a480a45"},  ]  colorama = [      {file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"}, @@ -901,7 +880,7 @@ django = [      {file = "Django-3.1.14.tar.gz", hash = "sha256:72a4a5a136a214c39cf016ccdd6b69e2aa08c7479c66d93f3a9b5e4bb9d8a347"},  ]  django-distill = [ -    {file = "django-distill-2.9.1.tar.gz", hash = "sha256:d849f84f6b763c26980d6bbdecfb1bd69859effef42704fee84f34fa57c82554"}, +    {file = "django-distill-2.9.2.tar.gz", hash = "sha256:91d5f45c2ff78b8efd4805ff5ec17df4ba815bbf51ca12a2cea65727d2f1d42e"},  ]  django-environ = [      {file = "django-environ-0.4.5.tar.gz", hash = "sha256:6c9d87660142608f63ec7d5ce5564c49b603ea8ff25da595fd6098f6dc82afde"}, @@ -912,8 +891,8 @@ django-filter = [      {file = "django_filter-21.1-py3-none-any.whl", hash = "sha256:f4a6737a30104c98d2e2a5fb93043f36dd7978e0c7ddc92f5998e85433ea5063"},  ]  django-prometheus = [ -    {file = "django-prometheus-2.1.0.tar.gz", hash = "sha256:dd3f8da1399140fbef5c00d1526a23d1ade286b144281c325f8e409a781643f2"}, -    {file = "django_prometheus-2.1.0-py2.py3-none-any.whl", hash = "sha256:c338d6efde1ca336e90c540b5e87afe9287d7bcc82d651a778f302b0be17a933"}, +    {file = "django-prometheus-2.2.0.tar.gz", hash = "sha256:240378a1307c408bd5fc85614a3a57f1ce633d4a222c9e291e2bbf325173b801"}, +    {file = "django_prometheus-2.2.0-py2.py3-none-any.whl", hash = "sha256:e6616770d8820b8834762764bf1b76ec08e1b98e72a6f359d488a2e15fe3537c"},  ]  django-simple-bulma = [      {file = "django-simple-bulma-2.4.0.tar.gz", hash = "sha256:99a15261b0c61062a128af3c6a45da9c066d6a4a548c9063464e0fb7a5438aa1"}, @@ -927,8 +906,8 @@ docopt = [      {file = "docopt-0.6.2.tar.gz", hash = "sha256:49b3a825280bd66b3aa83585ef59c4a8c82f2c8a522dbe754a8bc8d08c85c491"},  ]  filelock = [ -    {file = "filelock-3.4.0-py3-none-any.whl", hash = "sha256:2e139a228bcf56dd8b2274a65174d005c4a6b68540ee0bdbb92c76f43f29f7e8"}, -    {file = "filelock-3.4.0.tar.gz", hash = "sha256:93d512b32a23baf4cac44ffd72ccf70732aeff7b8050fcaf6d3ec406d954baf4"}, +    {file = "filelock-3.4.2-py3-none-any.whl", hash = "sha256:cf0fc6a2f8d26bd900f19bf33915ca70ba4dd8c56903eeb14e1e7a2fd7590146"}, +    {file = "filelock-3.4.2.tar.gz", hash = "sha256:38b4f4c989f9d06d44524df1b24bd19e167d851f19b50bf3e3559952dddc5b80"},  ]  flake8 = [      {file = "flake8-3.9.2-py2.py3-none-any.whl", hash = "sha256:bf8fd333346d844f616e8d47905ef3a3384edae6b4e9beb0c5101e25e3110907"}, @@ -962,8 +941,8 @@ flake8-string-format = [      {file = "flake8_string_format-0.3.0-py2.py3-none-any.whl", hash = "sha256:812ff431f10576a74c89be4e85b8e075a705be39bc40c4b4278b5b13e2afa9af"},  ]  flake8-tidy-imports = [ -    {file = "flake8-tidy-imports-4.5.0.tar.gz", hash = "sha256:ac637961d0f319012d099e49619f8c928e3221f74e00fe6eb89513bc64c40adb"}, -    {file = "flake8_tidy_imports-4.5.0-py3-none-any.whl", hash = "sha256:87eed94ae6a2fda6a5918d109746feadf1311e0eb8274ab7a7920f6db00a41c9"}, +    {file = "flake8-tidy-imports-4.6.0.tar.gz", hash = "sha256:3e193d8c4bb4492408a90e956d888b27eed14c698387c9b38230da3dad78058f"}, +    {file = "flake8_tidy_imports-4.6.0-py3-none-any.whl", hash = "sha256:6ae9f55d628156e19d19f4c359dd5d3e95431a9bd514f5e2748c53c1398c66b2"},  ]  flake8-todo = [      {file = "flake8-todo-0.7.tar.gz", hash = "sha256:6e4c5491ff838c06fe5a771b0e95ee15fc005ca57196011011280fc834a85915"}, @@ -973,24 +952,24 @@ gitdb = [      {file = "gitdb-4.0.9.tar.gz", hash = "sha256:bac2fd45c0a1c9cf619e63a90d62bdc63892ef92387424b855792a6cabe789aa"},  ]  gitpython = [ -    {file = "GitPython-3.1.24-py3-none-any.whl", hash = "sha256:dc0a7f2f697657acc8d7f89033e8b1ea94dd90356b2983bca89dc8d2ab3cc647"}, -    {file = "GitPython-3.1.24.tar.gz", hash = "sha256:df83fdf5e684fef7c6ee2c02fc68a5ceb7e7e759d08b694088d0cacb4eba59e5"}, +    {file = "GitPython-3.1.26-py3-none-any.whl", hash = "sha256:26ac35c212d1f7b16036361ca5cff3ec66e11753a0d677fb6c48fa4e1a9dd8d6"}, +    {file = "GitPython-3.1.26.tar.gz", hash = "sha256:fc8868f63a2e6d268fb25f481995ba185a85a66fcad126f039323ff6635669ee"},  ]  gunicorn = [      {file = "gunicorn-20.0.4-py2.py3-none-any.whl", hash = "sha256:cd4a810dd51bf497552cf3f863b575dabd73d6ad6a91075b65936b151cbf4f9c"},      {file = "gunicorn-20.0.4.tar.gz", hash = "sha256:1904bb2b8a43658807108d59c3f3d56c2b6121a701161de0ddf9ad140073c626"},  ]  identify = [ -    {file = "identify-2.4.0-py2.py3-none-any.whl", hash = "sha256:eba31ca80258de6bb51453084bff4a923187cd2193b9c13710f2516ab30732cc"}, -    {file = "identify-2.4.0.tar.gz", hash = "sha256:a33ae873287e81651c7800ca309dc1f84679b763c9c8b30680e16fbfa82f0107"}, +    {file = "identify-2.4.6-py2.py3-none-any.whl", hash = "sha256:cf06b1639e0dca0c184b1504d8b73448c99a68e004a80524c7923b95f7b6837c"}, +    {file = "identify-2.4.6.tar.gz", hash = "sha256:233679e3f61a02015d4293dbccf16aa0e4996f868bd114688b8c124f18826706"},  ]  idna = [      {file = "idna-3.3-py3-none-any.whl", hash = "sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff"},      {file = "idna-3.3.tar.gz", hash = "sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d"},  ]  importlib-metadata = [ -    {file = "importlib_metadata-4.8.2-py3-none-any.whl", hash = "sha256:53ccfd5c134223e497627b9815d5030edf77d2ed573922f7a0b8f8bb81a1c100"}, -    {file = "importlib_metadata-4.8.2.tar.gz", hash = "sha256:75bdec14c397f528724c1bfd9709d660b33a4d2e77387a3358f20b848bb5e5fb"}, +    {file = "importlib_metadata-4.10.1-py3-none-any.whl", hash = "sha256:899e2a40a8c4a1aec681feef45733de8a6c58f3f6a0dbed2eb6574b4387a77b6"}, +    {file = "importlib_metadata-4.10.1.tar.gz", hash = "sha256:951f0d8a5b7260e9db5e41d429285b5f451e928479f19d80818878527d36e95e"},  ]  libsass = [      {file = "libsass-0.21.0-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:06c8776417fe930714bdc930a3d7e795ae3d72be6ac883ff72a1b8f7c49e5ffb"}, @@ -1029,46 +1008,50 @@ pep8-naming = [      {file = "pep8_naming-0.12.1-py2.py3-none-any.whl", hash = "sha256:4a8daeaeb33cfcde779309fc0c9c0a68a3bbe2ad8a8308b763c5068f86eb9f37"},  ]  platformdirs = [ -    {file = "platformdirs-2.4.0-py3-none-any.whl", hash = "sha256:8868bbe3c3c80d42f20156f22e7131d2fb321f5bc86a2a345375c6481a67021d"}, -    {file = "platformdirs-2.4.0.tar.gz", hash = "sha256:367a5e80b3d04d2428ffa76d33f124cf11e8fff2acdaa9b43d545f5c7d661ef2"}, +    {file = "platformdirs-2.4.1-py3-none-any.whl", hash = "sha256:1d7385c7db91728b83efd0ca99a5afb296cab9d0ed8313a45ed8ba17967ecfca"}, +    {file = "platformdirs-2.4.1.tar.gz", hash = "sha256:440633ddfebcc36264232365d7840a970e75e1018d15b4327d11f91909045fda"},  ]  pre-commit = [ -    {file = "pre_commit-2.16.0-py2.py3-none-any.whl", hash = "sha256:758d1dc9b62c2ed8881585c254976d66eae0889919ab9b859064fc2fe3c7743e"}, -    {file = "pre_commit-2.16.0.tar.gz", hash = "sha256:fe9897cac830aa7164dbd02a4e7b90cae49630451ce88464bca73db486ba9f65"}, +    {file = "pre_commit-2.17.0-py2.py3-none-any.whl", hash = "sha256:725fa7459782d7bec5ead072810e47351de01709be838c2ce1726b9591dad616"}, +    {file = "pre_commit-2.17.0.tar.gz", hash = "sha256:c1a8040ff15ad3d648c70cc3e55b93e4d2d5b687320955505587fd79bbaed06a"},  ]  prometheus-client = [ -    {file = "prometheus_client-0.12.0-py2.py3-none-any.whl", hash = "sha256:317453ebabff0a1b02df7f708efbab21e3489e7072b61cb6957230dd004a0af0"}, -    {file = "prometheus_client-0.12.0.tar.gz", hash = "sha256:1b12ba48cee33b9b0b9de64a1047cbd3c5f2d0ab6ebcead7ddda613a750ec3c5"}, +    {file = "prometheus_client-0.13.1-py3-none-any.whl", hash = "sha256:357a447fd2359b0a1d2e9b311a0c5778c330cfbe186d880ad5a6b39884652316"}, +    {file = "prometheus_client-0.13.1.tar.gz", hash = "sha256:ada41b891b79fca5638bd5cfe149efa86512eaa55987893becd2c6d8d0a5dfc5"},  ]  psutil = [ -    {file = "psutil-5.8.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:0066a82f7b1b37d334e68697faba68e5ad5e858279fd6351c8ca6024e8d6ba64"}, -    {file = "psutil-5.8.0-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:0ae6f386d8d297177fd288be6e8d1afc05966878704dad9847719650e44fc49c"}, -    {file = "psutil-5.8.0-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:12d844996d6c2b1d3881cfa6fa201fd635971869a9da945cf6756105af73d2df"}, -    {file = "psutil-5.8.0-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:02b8292609b1f7fcb34173b25e48d0da8667bc85f81d7476584d889c6e0f2131"}, -    {file = "psutil-5.8.0-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:6ffe81843131ee0ffa02c317186ed1e759a145267d54fdef1bc4ea5f5931ab60"}, -    {file = "psutil-5.8.0-cp27-none-win32.whl", hash = "sha256:ea313bb02e5e25224e518e4352af4bf5e062755160f77e4b1767dd5ccb65f876"}, -    {file = "psutil-5.8.0-cp27-none-win_amd64.whl", hash = "sha256:5da29e394bdedd9144c7331192e20c1f79283fb03b06e6abd3a8ae45ffecee65"}, -    {file = "psutil-5.8.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:74fb2557d1430fff18ff0d72613c5ca30c45cdbfcddd6a5773e9fc1fe9364be8"}, -    {file = "psutil-5.8.0-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:74f2d0be88db96ada78756cb3a3e1b107ce8ab79f65aa885f76d7664e56928f6"}, -    {file = "psutil-5.8.0-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:99de3e8739258b3c3e8669cb9757c9a861b2a25ad0955f8e53ac662d66de61ac"}, -    {file = "psutil-5.8.0-cp36-cp36m-win32.whl", hash = "sha256:36b3b6c9e2a34b7d7fbae330a85bf72c30b1c827a4366a07443fc4b6270449e2"}, -    {file = "psutil-5.8.0-cp36-cp36m-win_amd64.whl", hash = "sha256:52de075468cd394ac98c66f9ca33b2f54ae1d9bff1ef6b67a212ee8f639ec06d"}, -    {file = "psutil-5.8.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:c6a5fd10ce6b6344e616cf01cc5b849fa8103fbb5ba507b6b2dee4c11e84c935"}, -    {file = "psutil-5.8.0-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:61f05864b42fedc0771d6d8e49c35f07efd209ade09a5afe6a5059e7bb7bf83d"}, -    {file = "psutil-5.8.0-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:0dd4465a039d343925cdc29023bb6960ccf4e74a65ad53e768403746a9207023"}, -    {file = "psutil-5.8.0-cp37-cp37m-win32.whl", hash = "sha256:1bff0d07e76114ec24ee32e7f7f8d0c4b0514b3fae93e3d2aaafd65d22502394"}, -    {file = "psutil-5.8.0-cp37-cp37m-win_amd64.whl", hash = "sha256:fcc01e900c1d7bee2a37e5d6e4f9194760a93597c97fee89c4ae51701de03563"}, -    {file = "psutil-5.8.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6223d07a1ae93f86451d0198a0c361032c4c93ebd4bf6d25e2fb3edfad9571ef"}, -    {file = "psutil-5.8.0-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:d225cd8319aa1d3c85bf195c4e07d17d3cd68636b8fc97e6cf198f782f99af28"}, -    {file = "psutil-5.8.0-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:28ff7c95293ae74bf1ca1a79e8805fcde005c18a122ca983abf676ea3466362b"}, -    {file = "psutil-5.8.0-cp38-cp38-win32.whl", hash = "sha256:ce8b867423291cb65cfc6d9c4955ee9bfc1e21fe03bb50e177f2b957f1c2469d"}, -    {file = "psutil-5.8.0-cp38-cp38-win_amd64.whl", hash = "sha256:90f31c34d25b1b3ed6c40cdd34ff122b1887a825297c017e4cbd6796dd8b672d"}, -    {file = "psutil-5.8.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6323d5d845c2785efb20aded4726636546b26d3b577aded22492908f7c1bdda7"}, -    {file = "psutil-5.8.0-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:245b5509968ac0bd179287d91210cd3f37add77dad385ef238b275bad35fa1c4"}, -    {file = "psutil-5.8.0-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:90d4091c2d30ddd0a03e0b97e6a33a48628469b99585e2ad6bf21f17423b112b"}, -    {file = "psutil-5.8.0-cp39-cp39-win32.whl", hash = "sha256:ea372bcc129394485824ae3e3ddabe67dc0b118d262c568b4d2602a7070afdb0"}, -    {file = "psutil-5.8.0-cp39-cp39-win_amd64.whl", hash = "sha256:f4634b033faf0d968bb9220dd1c793b897ab7f1189956e1aa9eae752527127d3"}, -    {file = "psutil-5.8.0.tar.gz", hash = "sha256:0c9ccb99ab76025f2f0bbecf341d4656e9c1351db8cc8a03ccd62e318ab4b5c6"}, +    {file = "psutil-5.9.0-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:55ce319452e3d139e25d6c3f85a1acf12d1607ddedea5e35fb47a552c051161b"}, +    {file = "psutil-5.9.0-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:7336292a13a80eb93c21f36bde4328aa748a04b68c13d01dfddd67fc13fd0618"}, +    {file = "psutil-5.9.0-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:cb8d10461c1ceee0c25a64f2dd54872b70b89c26419e147a05a10b753ad36ec2"}, +    {file = "psutil-5.9.0-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:7641300de73e4909e5d148e90cc3142fb890079e1525a840cf0dfd39195239fd"}, +    {file = "psutil-5.9.0-cp27-none-win32.whl", hash = "sha256:ea42d747c5f71b5ccaa6897b216a7dadb9f52c72a0fe2b872ef7d3e1eacf3ba3"}, +    {file = "psutil-5.9.0-cp27-none-win_amd64.whl", hash = "sha256:ef216cc9feb60634bda2f341a9559ac594e2eeaadd0ba187a4c2eb5b5d40b91c"}, +    {file = "psutil-5.9.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:90a58b9fcae2dbfe4ba852b57bd4a1dded6b990a33d6428c7614b7d48eccb492"}, +    {file = "psutil-5.9.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ff0d41f8b3e9ebb6b6110057e40019a432e96aae2008951121ba4e56040b84f3"}, +    {file = "psutil-5.9.0-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:742c34fff804f34f62659279ed5c5b723bb0195e9d7bd9907591de9f8f6558e2"}, +    {file = "psutil-5.9.0-cp310-cp310-win32.whl", hash = "sha256:8293942e4ce0c5689821f65ce6522ce4786d02af57f13c0195b40e1edb1db61d"}, +    {file = "psutil-5.9.0-cp310-cp310-win_amd64.whl", hash = "sha256:9b51917c1af3fa35a3f2dabd7ba96a2a4f19df3dec911da73875e1edaf22a40b"}, +    {file = "psutil-5.9.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:e9805fed4f2a81de98ae5fe38b75a74c6e6ad2df8a5c479594c7629a1fe35f56"}, +    {file = "psutil-5.9.0-cp36-cp36m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c51f1af02334e4b516ec221ee26b8fdf105032418ca5a5ab9737e8c87dafe203"}, +    {file = "psutil-5.9.0-cp36-cp36m-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:32acf55cb9a8cbfb29167cd005951df81b567099295291bcfd1027365b36591d"}, +    {file = "psutil-5.9.0-cp36-cp36m-win32.whl", hash = "sha256:e5c783d0b1ad6ca8a5d3e7b680468c9c926b804be83a3a8e95141b05c39c9f64"}, +    {file = "psutil-5.9.0-cp36-cp36m-win_amd64.whl", hash = "sha256:d62a2796e08dd024b8179bd441cb714e0f81226c352c802fca0fd3f89eeacd94"}, +    {file = "psutil-5.9.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:3d00a664e31921009a84367266b35ba0aac04a2a6cad09c550a89041034d19a0"}, +    {file = "psutil-5.9.0-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7779be4025c540d1d65a2de3f30caeacc49ae7a2152108adeaf42c7534a115ce"}, +    {file = "psutil-5.9.0-cp37-cp37m-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:072664401ae6e7c1bfb878c65d7282d4b4391f1bc9a56d5e03b5a490403271b5"}, +    {file = "psutil-5.9.0-cp37-cp37m-win32.whl", hash = "sha256:df2c8bd48fb83a8408c8390b143c6a6fa10cb1a674ca664954de193fdcab36a9"}, +    {file = "psutil-5.9.0-cp37-cp37m-win_amd64.whl", hash = "sha256:1d7b433519b9a38192dfda962dd8f44446668c009833e1429a52424624f408b4"}, +    {file = "psutil-5.9.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:c3400cae15bdb449d518545cbd5b649117de54e3596ded84aacabfbb3297ead2"}, +    {file = "psutil-5.9.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b2237f35c4bbae932ee98902a08050a27821f8f6dfa880a47195e5993af4702d"}, +    {file = "psutil-5.9.0-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1070a9b287846a21a5d572d6dddd369517510b68710fca56b0e9e02fd24bed9a"}, +    {file = "psutil-5.9.0-cp38-cp38-win32.whl", hash = "sha256:76cebf84aac1d6da5b63df11fe0d377b46b7b500d892284068bacccf12f20666"}, +    {file = "psutil-5.9.0-cp38-cp38-win_amd64.whl", hash = "sha256:3151a58f0fbd8942ba94f7c31c7e6b310d2989f4da74fcbf28b934374e9bf841"}, +    {file = "psutil-5.9.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:539e429da49c5d27d5a58e3563886057f8fc3868a5547b4f1876d9c0f007bccf"}, +    {file = "psutil-5.9.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:58c7d923dc209225600aec73aa2c4ae8ea33b1ab31bc11ef8a5933b027476f07"}, +    {file = "psutil-5.9.0-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3611e87eea393f779a35b192b46a164b1d01167c9d323dda9b1e527ea69d697d"}, +    {file = "psutil-5.9.0-cp39-cp39-win32.whl", hash = "sha256:4e2fb92e3aeae3ec3b7b66c528981fd327fb93fd906a77215200404444ec1845"}, +    {file = "psutil-5.9.0-cp39-cp39-win_amd64.whl", hash = "sha256:7d190ee2eaef7831163f254dc58f6d2e2a22e27382b936aab51c835fc080c3d3"}, +    {file = "psutil-5.9.0.tar.gz", hash = "sha256:869842dbd66bb80c3217158e629d6fceaecc3a3166d3d1faee515b05dd26ca25"},  ]  psycopg2-binary = [      {file = "psycopg2-binary-2.8.6.tar.gz", hash = "sha256:11b9c0ebce097180129e422379b824ae21c8f2a6596b159c7659e2e5a00e1aa0"}, @@ -1167,8 +1150,8 @@ pyyaml = [      {file = "PyYAML-5.4.1.tar.gz", hash = "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e"},  ]  requests = [ -    {file = "requests-2.26.0-py2.py3-none-any.whl", hash = "sha256:6c1246513ecd5ecd4528a0906f910e8f0f9c6b8ec72030dc9fd154dc1a6efd24"}, -    {file = "requests-2.26.0.tar.gz", hash = "sha256:b8aa58f8cf793ffd8782d3d8cb19e66ef36f7aba4353eec859e74678b01b07a7"}, +    {file = "requests-2.27.1-py2.py3-none-any.whl", hash = "sha256:f22fa1e554c9ddfd16e6e41ac79759e17be9e492b3587efa038054674760e72d"}, +    {file = "requests-2.27.1.tar.gz", hash = "sha256:68d7c56fd5a8999887728ef304a6d12edc7be74f1cfa47714fc8b414525c9a61"},  ]  sentry-sdk = [      {file = "sentry-sdk-0.20.3.tar.gz", hash = "sha256:4ae8d1ced6c67f1c8ea51d82a16721c166c489b76876c9f2c202b8a50334b237"}, @@ -1202,23 +1185,19 @@ toml = [      {file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"},      {file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"},  ] -typing-extensions = [ -    {file = "typing_extensions-4.0.1-py3-none-any.whl", hash = "sha256:7f001e5ac290a0c0401508864c7ec868be4e701886d5b573a9528ed3973d9d3b"}, -    {file = "typing_extensions-4.0.1.tar.gz", hash = "sha256:4ca091dea149f945ec56afb48dae714f21e8692ef22a395223bcd328961b6a0e"}, -]  urllib3 = [ -    {file = "urllib3-1.26.7-py2.py3-none-any.whl", hash = "sha256:c4fdf4019605b6e5423637e01bc9fe4daef873709a7973e195ceba0a62bbc844"}, -    {file = "urllib3-1.26.7.tar.gz", hash = "sha256:4987c65554f7a2dbf30c18fd48778ef124af6fab771a377103da0585e2336ece"}, +    {file = "urllib3-1.26.8-py2.py3-none-any.whl", hash = "sha256:000ca7f471a233c2251c6c7023ee85305721bfdf18621ebff4fd17a8653427ed"}, +    {file = "urllib3-1.26.8.tar.gz", hash = "sha256:0e7c33d9a63e7ddfcb86780aac87befc2fbddf46c58dbb487e0855f7ceec283c"},  ]  virtualenv = [ -    {file = "virtualenv-20.10.0-py2.py3-none-any.whl", hash = "sha256:4b02e52a624336eece99c96e3ab7111f469c24ba226a53ec474e8e787b365814"}, -    {file = "virtualenv-20.10.0.tar.gz", hash = "sha256:576d05b46eace16a9c348085f7d0dc8ef28713a2cabaa1cf0aea41e8f12c9218"}, +    {file = "virtualenv-20.13.0-py2.py3-none-any.whl", hash = "sha256:339f16c4a86b44240ba7223d0f93a7887c3ca04b5f9c8129da7958447d079b09"}, +    {file = "virtualenv-20.13.0.tar.gz", hash = "sha256:d8458cf8d59d0ea495ad9b34c2599487f8a7772d796f9910858376d1600dd2dd"},  ]  whitenoise = [      {file = "whitenoise-5.3.0-py2.py3-none-any.whl", hash = "sha256:d963ef25639d1417e8a247be36e6aedd8c7c6f0a08adcb5a89146980a96b577c"},      {file = "whitenoise-5.3.0.tar.gz", hash = "sha256:d234b871b52271ae7ed6d9da47ffe857c76568f11dd30e28e18c5869dbd11e12"},  ]  zipp = [ -    {file = "zipp-3.6.0-py3-none-any.whl", hash = "sha256:9fe5ea21568a0a70e50f273397638d39b03353731e6cbbb3fd8502a33fec40bc"}, -    {file = "zipp-3.6.0.tar.gz", hash = "sha256:71c644c5369f4a6e07636f0aa966270449561fcea2e3d6747b8d23efaa9d7832"}, +    {file = "zipp-3.7.0-py3-none-any.whl", hash = "sha256:b47250dd24f92b7dd6a0a8fc5244da14608f3ca90a5efcd37a3b1642fac9a375"}, +    {file = "zipp-3.7.0.tar.gz", hash = "sha256:9f50f446828eb9d45b267433fd3e9da8d801f614129124863f9c51ebceafb87d"},  ] diff --git a/pydis_site/apps/api/models/bot/metricity.py b/pydis_site/apps/api/models/bot/metricity.py index 901f191a..abd25ef0 100644 --- a/pydis_site/apps/api/models/bot/metricity.py +++ b/pydis_site/apps/api/models/bot/metricity.py @@ -10,7 +10,7 @@ EXCLUDE_CHANNELS = (  ) -class NotFoundError(Exception): +class NotFoundError(Exception):  # noqa: N818      """Raised when an entity cannot be found."""      pass diff --git a/pydis_site/apps/content/resources/guides/pydis-guides/contributing/bot.md b/pydis_site/apps/content/resources/guides/pydis-guides/contributing/bot.md index ed9e3db3..2aa10aa3 100644 --- a/pydis_site/apps/content/resources/guides/pydis-guides/contributing/bot.md +++ b/pydis_site/apps/content/resources/guides/pydis-guides/contributing/bot.md @@ -14,10 +14,10 @@ First things first, to run the bot's code and make changes to it, you need a loc      <button type="button" class="card-header collapsible">          <span class="card-header-title subtitle is-6 my-2 ml-2">Getting started with Git and GitHub</span>          <span class="card-header-icon"> -            <i class="fas fa-angle-down title is-5" aria-hidden="true"></i> +            <i class="fas fa-fw fa-angle-down title is-5" aria-hidden="true"></i>          </span>      </button> -    <div class="collapsible-content"> +    <div class="collapsible-content collapsed">          <div class="card-content">                <p>If you don't have Git on your computer already, <a href="https://git-scm.com/downloads">install it</a>. You can additionally install a Git GUI such as <a href="https://www.gitkraken.com/download">GitKraken</a>, or the <a href="https://cli.github.com/manual/installation">GitHub CLI</a>.</p>                <p>To learn more about Git, you can look into <a href="../working-with-git">our guides</a>, as well as <a href="https://education.github.com/git-cheat-sheet-education.pdf">this cheatsheet</a>, <a href="https://learngitbranching.js.org">Learn Git Branching</a>, and otherwise any guide you can find on the internet. Once you got the basic idea though, the best way to learn Git is to use it.</p> @@ -78,10 +78,10 @@ See [here](../obtaining-discord-ids) for help with obtaining Discord IDs.      <button type="button" class="card-header collapsible">          <span class="card-header-title subtitle is-6 my-2 ml-2">Optional config.yml</span>          <span class="card-header-icon"> -            <i class="fas fa-angle-down title is-5" aria-hidden="true"></i> +            <i class="fas fa-fw fa-angle-down title is-5" aria-hidden="true"></i>          </span>      </button> -    <div class="collapsible-content"> +    <div class="collapsible-content collapsed">          <div class="card-content">                <p>If you used the provided server template, and you're not sure which channels belong where in the config file, you can use the config below. Pay attention to the comments with several <code>#</code> symbols, and replace the <code>�</code> characters with the right IDs.</p>                <pre> @@ -458,10 +458,10 @@ We understand this is tedious and are working on a better solution for setting u      <button type="button" class="card-header collapsible">          <span class="card-header-title subtitle is-6 my-2 ml-2">Why do you need a separate config file?</span>          <span class="card-header-icon"> -            <i class="fas fa-angle-down title is-5" aria-hidden="true"></i> +            <i class="fas fa-fw fa-angle-down title is-5" aria-hidden="true"></i>          </span>      </button> -    <div class="collapsible-content"> +    <div class="collapsible-content collapsed">          <div class="card-content">              While it's technically possible to edit <code>config-default.yml</code> to match your server, it is heavily discouraged.              This file's purpose is to provide the configurations the Python bot needs to run in the Python server in production, and should remain as such. @@ -487,10 +487,10 @@ You are now almost ready to run the Python bot. The simplest way to do so is wit      <button type="button" class="card-header collapsible">          <span class="card-header-title subtitle is-6 my-2 ml-2">Getting started with Docker</span>          <span class="card-header-icon"> -            <i class="fas fa-angle-down title is-5" aria-hidden="true"></i> +            <i class="fas fa-fw fa-angle-down title is-5" aria-hidden="true"></i>          </span>      </button> -    <div class="collapsible-content"> +    <div class="collapsible-content collapsed">          <div class="card-content">              The requirements for Docker are:              <ul> @@ -541,10 +541,10 @@ With at least the site running in Docker already (see the previous section on ho      <button type="button" class="card-header collapsible">          <span class="card-header-title subtitle is-6 my-2 ml-2">Ways to run code</span>          <span class="card-header-icon"> -            <i class="fas fa-angle-down title is-5" aria-hidden="true"></i> +            <i class="fas fa-fw fa-angle-down title is-5" aria-hidden="true"></i>          </span>      </button> -    <div class="collapsible-content"> +    <div class="collapsible-content collapsed">          <div class="card-content">              Notice that the bot is started as a module. There are several ways to do so:              <ul> diff --git a/pydis_site/apps/content/urls.py b/pydis_site/apps/content/urls.py index fe7c2852..f8496095 100644 --- a/pydis_site/apps/content/urls.py +++ b/pydis_site/apps/content/urls.py @@ -30,7 +30,7 @@ def __get_all_files(root: Path, folder: typing.Optional[Path] = None) -> list[st  def get_all_pages() -> typing.Iterator[dict[str, str]]: -    """Yield a dict of all pag categories.""" +    """Yield a dict of all page categories."""      for location in __get_all_files(Path("pydis_site", "apps", "content", "resources")):          yield {"location": location} diff --git a/pydis_site/apps/redirect/redirects.yaml b/pydis_site/apps/redirect/redirects.yaml index 9bcf3afd..9b64011b 100644 --- a/pydis_site/apps/redirect/redirects.yaml +++ b/pydis_site/apps/redirect/redirects.yaml @@ -87,9 +87,45 @@ resources_index_redirect:    original_path: pages/resources/    redirect_route: "resources:index" -resources_resources_redirect: -  original_path: pages/resources/<str:category>/ -  redirect_route: "resources:resources" +resources_reading_redirect: +  original_path: resources/reading/ +  redirect_route: "resources:index" +  redirect_arguments: ["book"] + +resources_books_redirect: +  original_path: resources/books/ +  redirect_route: "resources:index" +  redirect_arguments: ["book"] + +resources_videos_redirect: +  original_path: resources/videos/ +  redirect_route: "resources:index" +  redirect_arguments: ["video"] + +resources_courses_redirect: +  original_path: resources/courses/ +  redirect_route: "resources:index" +  redirect_arguments: ["course"] + +resources_communities_redirect: +  original_path: resources/communities/ +  redirect_route: "resources:index" +  redirect_arguments: ["community"] + +resources_podcasts_redirect: +  original_path: resources/podcasts/ +  redirect_route: "resources:index" +  redirect_arguments: ["podcast"] + +resources_tutorials_redirect: +  original_path: resources/tutorials/ +  redirect_route: "resources:index" +  redirect_arguments: ["tutorial"] + +resources_tools_redirect: +  original_path: resources/tools/ +  redirect_route: "resources:index" +  redirect_arguments: ["tool"]  # Events  events_index_redirect: diff --git a/pydis_site/apps/resources/resources/communities/adafruit.yaml b/pydis_site/apps/resources/resources/adafruit.yaml index e5c81a6c..f9466bd8 100644 --- a/pydis_site/apps/resources/resources/communities/adafruit.yaml +++ b/pydis_site/apps/resources/resources/adafruit.yaml @@ -4,12 +4,18 @@ description: Adafruit is an open-source electronics manufacturer    provide help with your projects,    and the Adafruit devs do all the CircuitPython Development right out in the open.  title_image: https://www.mouser.com/images/suppliers/logos/adafruit.png -title_url: https://discord.gg/adafruit -position: 4 +title_url: https://adafruit.com/  urls: -- icon: branding/discord -  url: https://discord.gg/adafruit -  color: blurple -- icon: regular/link -  url: https://adafruit.com/ -  color: teal +  - icon: branding/discord +    url: https://discord.gg/adafruit +    color: blurple +tags: +  topics: +    - microcontrollers +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/tools/editors/atom.yaml b/pydis_site/apps/resources/resources/atom.yaml index c44f9b5b..26e125b1 100644 --- a/pydis_site/apps/resources/resources/tools/editors/atom.yaml +++ b/pydis_site/apps/resources/resources/atom.yaml @@ -2,4 +2,13 @@ description: A free Electron-based editor, a "hackable text editor for the 21st    by the GitHub team.  name: Atom  title_url: https://atom.io/ -position: 0 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/reading/books/automate_the_boring_stuff.yaml b/pydis_site/apps/resources/resources/automate_the_boring_stuff_book.yaml index 3812029c..63f63193 100644 --- a/pydis_site/apps/resources/resources/reading/books/automate_the_boring_stuff.yaml +++ b/pydis_site/apps/resources/resources/automate_the_boring_stuff_book.yaml @@ -4,11 +4,18 @@ description: One of the best books out there for Python beginners. This book wil    the web, manipulating files and automating keyboard and mouse input. Ideal for an    office worker who wants to make himself more useful.  name: Automate the Boring Stuff with Python -position: 2 +title_url: https://automatetheboringstuff.com/  urls: -- icon: regular/book -  url: https://automatetheboringstuff.com/ +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/22514127-automate-the-boring-stuff-with-python    color: black -- icon: branding/amazon -  url: https://www.amazon.com/Automate-Boring-Stuff-Python-Programming/dp/1593275994/ -  color: amazon-orange +tags: +  topics: +    - general +  payment_tiers: +    - free +    - paid +  difficulty: +    - beginner +  type: +    - book diff --git a/pydis_site/apps/resources/resources/courses/automate_the_boring_stuff_with_python.yaml b/pydis_site/apps/resources/resources/automate_the_boring_stuff_course.yaml index 66034ea2..133033f7 100644 --- a/pydis_site/apps/resources/resources/courses/automate_the_boring_stuff_with_python.yaml +++ b/pydis_site/apps/resources/resources/automate_the_boring_stuff_course.yaml @@ -2,4 +2,13 @@ description: The interactive course version of Al Sweigart's excellent book for    This link has a discounted version of the course which will always cost 10 dollars. Thanks, Al!  name: Automate the Boring Stuff with Python  title_url: https://www.udemy.com/automate/?couponCode=FOR_LIKE_10_BUCKS -position: 3 +tags: +  topics: +    - general +  payment_tiers: +    - paid +  difficulty: +    - beginner +  type: +    - course +    - interactive diff --git a/pydis_site/apps/resources/resources/communities/awesome_programming_discord.yaml b/pydis_site/apps/resources/resources/awesome_programming_discord.yaml index 335ac507..0ef7aefc 100644 --- a/pydis_site/apps/resources/resources/communities/awesome_programming_discord.yaml +++ b/pydis_site/apps/resources/resources/awesome_programming_discord.yaml @@ -6,4 +6,13 @@ title_icon: branding/github  title_icon_color: black  title_url: https://github.com/mhxion/awesome-programming-discord  name: awesome-programming-discord -position: 10 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/reading/books/byte_of_python.yaml b/pydis_site/apps/resources/resources/byte_of_python.yaml index 2530c1a4..c2f6ab84 100644 --- a/pydis_site/apps/resources/resources/reading/books/byte_of_python.yaml +++ b/pydis_site/apps/resources/resources/byte_of_python.yaml @@ -2,14 +2,21 @@ description: A free book on programming using the Python language.    It serves as a tutorial or guide to the Python language for a beginner audience.    If all you know about computers is how to save text files, then this is the book for you.  name: A Byte of Python -position: 1 +title_url: https://python.swaroopch.com/  urls: -- icon: regular/link -  url: https://python.swaroopch.com/ -  color: teal  - icon: regular/book    url: https://www.lulu.com/shop/swaroop-c-h/a-byte-of-python/paperback/product-21142968.html    color: black -- icon: branding/amazon -  url: https://www.amazon.com/Byte-Python-Swaroop-C-H-ebook/dp/B00FJ7S2JU/ -  color: amazon-orange +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/6762544-a-byte-of-python +  color: black +tags: +  topics: +    - general +  payment_tiers: +    - free +    - paid +  difficulty: +    - beginner +  type: +    - book diff --git a/pydis_site/apps/resources/resources/interactive/code_combat.yaml b/pydis_site/apps/resources/resources/code_combat.yaml index 30f20c28..84597c4d 100644 --- a/pydis_site/apps/resources/resources/interactive/code_combat.yaml +++ b/pydis_site/apps/resources/resources/code_combat.yaml @@ -1,11 +1,20 @@  description: Learn Python while gaming - an open-source project with thousands of    contributors, which teaches you Python through a deep, top-down RPG.  name: Code Combat -position: 0 +title_url: https://codecombat.com/  urls: -- icon: regular/link -  url: https://codecombat.com/ -  color: teal  - icon: branding/github    url: https://github.com/codecombat/codecombat    color: black +tags: +  topics: +    - general +    - algorithms and data structures +  payment_tiers: +    - free +    - subscription +  difficulty: +    - beginner +    - intermediate +  type: +    - interactive diff --git a/pydis_site/apps/resources/resources/communities/_category_info.yaml b/pydis_site/apps/resources/resources/communities/_category_info.yaml deleted file mode 100644 index b9cb6533..00000000 --- a/pydis_site/apps/resources/resources/communities/_category_info.yaml +++ /dev/null @@ -1,2 +0,0 @@ -description: Partnered communities that share part of our mission. -name: Communities diff --git a/pydis_site/apps/resources/resources/communities/django.yaml b/pydis_site/apps/resources/resources/communities/django.yaml deleted file mode 100644 index 0ef729ba..00000000 --- a/pydis_site/apps/resources/resources/communities/django.yaml +++ /dev/null @@ -1,13 +0,0 @@ -description: Django is a high-level Python web framework that encourages rapid development and clean, -  pragmatic design. Built by experienced developers, it takes care of much of the hassle of web development, -  so you can focus on writing your app without needing to reinvent the wheel. It’s free and open source. -title_image: https://static.djangoproject.com/img/logos/django-logo-positive.png -title_url: https://discord.gg/M9Mx3RvKM5 -position: 5 -urls: -  - icon: branding/discord -    url: https://discord.gg/M9Mx3RvKM5 -    color: blurple -  - icon: regular/link -    url: https://www.djangoproject.com/ -    color: teal diff --git a/pydis_site/apps/resources/resources/videos/corey_schafer.yaml b/pydis_site/apps/resources/resources/corey_schafer.yaml index a7cca18a..f5af2cab 100644 --- a/pydis_site/apps/resources/resources/videos/corey_schafer.yaml +++ b/pydis_site/apps/resources/resources/corey_schafer.yaml @@ -9,11 +9,21 @@ description: 'Corey has a number of exceptionally high quality tutorial series    Check out his channel for more video series!    '  title_image: https://i.imgur.com/KIfWw3b.png -position: 0 +title_url: https://www.youtube.com/channel/UCCezIgC97PvUuR4_gbFUs5g  urls: -  - icon: branding/youtube -    url: https://www.youtube.com/channel/UCCezIgC97PvUuR4_gbFUs5g -    color: youtube-red -  - icon: regular/link +  - icon: solid/external-link-alt      url: https://coreyms.com/      color: teal +tags: +  topics: +    - general +    - software design +    - web development +    - tooling +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - video diff --git a/pydis_site/apps/resources/resources/courses/_category_info.yaml b/pydis_site/apps/resources/resources/courses/_category_info.yaml deleted file mode 100644 index 948b48de..00000000 --- a/pydis_site/apps/resources/resources/courses/_category_info.yaml +++ /dev/null @@ -1,4 +0,0 @@ -description: Listing of best Python courses. -name: Courses -default_icon: regular/graduation-cap -default_icon_color: black diff --git a/pydis_site/apps/resources/resources/data_science_from_scratch.yaml b/pydis_site/apps/resources/resources/data_science_from_scratch.yaml new file mode 100644 index 00000000..86955fdb --- /dev/null +++ b/pydis_site/apps/resources/resources/data_science_from_scratch.yaml @@ -0,0 +1,22 @@ +description: Data Science from Scratch is a good introduction to data science for the complete beginner, and covers +  some of the fundamentals of Python programming as well as the basic math, probability and statistics needed to get +  started. While either edition of this book is useful for those with prior Python experience, complete beginners +  should use the second edition, which contains more up-to-date code examples and better practices. +name: Data Science from Scratch +title_url: https://www.oreilly.com/library/view/data-science-from/9781492041122/ +urls: +  - icon: branding/goodreads +    url: https://www.goodreads.com/en/book/show/52059715-data-science-from-scratch +    color: black +  - icon: branding/github +    url: https://github.com/joelgrus/data-science-from-scratch +    color: black +tags: +  topics: +    - data science +  payment_tiers: +    - paid +  difficulty: +    - beginner +  type: +    - book diff --git a/pydis_site/apps/resources/resources/interactive/edublocks.yaml b/pydis_site/apps/resources/resources/edublocks.yaml index 7c6ca02b..3eaefc35 100644 --- a/pydis_site/apps/resources/resources/interactive/edublocks.yaml +++ b/pydis_site/apps/resources/resources/edublocks.yaml @@ -7,4 +7,12 @@ description: EduBlocks provides a simple drag and drop interface to help beginne    and export the code to run on actual devices.  name: EduBlocks  title_url: https://edublocks.org/ -position: 5 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - interactive diff --git a/pydis_site/apps/resources/resources/effective_python.yaml b/pydis_site/apps/resources/resources/effective_python.yaml new file mode 100644 index 00000000..b82fa0c3 --- /dev/null +++ b/pydis_site/apps/resources/resources/effective_python.yaml @@ -0,0 +1,21 @@ +description: A book that gives 90 best practices for writing excellent Python. Great +  for intermediates. +name: Effective Python +title_url: https://effectivepython.com/ +urls: +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/48566725-effective-python +  color: black +- icon: branding/github +  url: https://github.com/bslatkin/effectivepython +  color: black +tags: +  topics: +    - general +    - software design +  payment_tiers: +    - paid +  difficulty: +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/interactive/exercism.yaml b/pydis_site/apps/resources/resources/exercism.yaml index 68b458d0..b8f53d72 100644 --- a/pydis_site/apps/resources/resources/interactive/exercism.yaml +++ b/pydis_site/apps/resources/resources/exercism.yaml @@ -3,11 +3,18 @@ description: Level up your programming skills with more than 2600 exercises acro    where you can get your code reviewed for each solution you submit. The mentors will    give you insightful advice to make you a better programmer.  name: exercism.io -position: 1 +title_url: https://exercism.io/  urls: -- icon: regular/link -  url: https://exercism.io/ -  color: teal  - icon: branding/github    url: https://github.com/exercism/python    color: black +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - interactive diff --git a/pydis_site/apps/resources/resources/flask_web_development.yaml b/pydis_site/apps/resources/resources/flask_web_development.yaml new file mode 100644 index 00000000..6905b2b4 --- /dev/null +++ b/pydis_site/apps/resources/resources/flask_web_development.yaml @@ -0,0 +1,21 @@ +description: A comprehensive Flask walkthrough that has you building a complete social +  blogging application from scratch. +name: Flask Web Development +title_url: http://shop.oreilly.com/product/0636920031116.do +urls: +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/18774655-flask-web-development +  color: black +- icon: branding/github +  url: https://github.com/miguelgrinberg/flasky +  color: black +tags: +  topics: +    - web development +  payment_tiers: +    - paid +  difficulty: +    - beginner +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/fluent_python.yaml b/pydis_site/apps/resources/resources/fluent_python.yaml new file mode 100644 index 00000000..c22fd388 --- /dev/null +++ b/pydis_site/apps/resources/resources/fluent_python.yaml @@ -0,0 +1,21 @@ +description: A veritable tome of intermediate and advanced Python information. A must-read +  for any Python professional. By far the most recommended book for intermediates. +name: Fluent Python +title_url: https://www.oreilly.com/library/view/fluent-python/9781491946237/ +urls: +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/22800567-fluent-python +  color: black +- icon: branding/github +  url: https://github.com/fluentpython +  color: black +tags: +  topics: +    - general +    - software design +  payment_tiers: +    - paid +  difficulty: +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/reading/tutorials/getting_started_with_kivy.yaml b/pydis_site/apps/resources/resources/getting_started_with_kivy.yaml index d1d9a7d2..06eb2c14 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/getting_started_with_kivy.yaml +++ b/pydis_site/apps/resources/resources/getting_started_with_kivy.yaml @@ -2,4 +2,13 @@ description: A big list of excellent resources for getting started making Kivy a  name: Getting Started with Kivy  title_url: https://blog.kivy.org/2019/12/getting-started-with-kivy/  icon_image: https://raw.githubusercontent.com/kivy/kivy-website/master/logos/kivy-logo-black-256.png -position: 3 +tags: +  topics: +    - user interface +    - game development +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/reading/tutorials/getting_started_with_python_for_non_programmers.yaml b/pydis_site/apps/resources/resources/getting_started_with_python_for_non_programmers.yaml index 3250a7c4..6fab0114 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/getting_started_with_python_for_non_programmers.yaml +++ b/pydis_site/apps/resources/resources/getting_started_with_python_for_non_programmers.yaml @@ -2,4 +2,12 @@ description: A list of beginner resources for programmers with no prior develope    from Python's official guide.  name: Getting Started with Python for Non-Programmers  title_url: https://wiki.python.org/moin/BeginnersGuide/NonProgrammers -position: 1 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/reading/tutorials/getting_started_with_python_for_programmers.yaml b/pydis_site/apps/resources/resources/getting_started_with_python_for_programmers.yaml index b65e0e12..74b6efb9 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/getting_started_with_python_for_programmers.yaml +++ b/pydis_site/apps/resources/resources/getting_started_with_python_for_programmers.yaml @@ -3,3 +3,12 @@ description: A list of beginner resources for programmers coming from other lang  name: Getting Started with Python for Programmers  title_url: https://wiki.python.org/moin/BeginnersGuide/Programmers  position: 0 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - intermediate +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/google_colab.yaml b/pydis_site/apps/resources/resources/google_colab.yaml new file mode 100644 index 00000000..b8936674 --- /dev/null +++ b/pydis_site/apps/resources/resources/google_colab.yaml @@ -0,0 +1,17 @@ +description: Google Colab is a high-powered custom version of Jupyter Notebook which supports e.g. +  !apt-get to install arbitrary Debian packages to the runtime, which is very generous with CPU and memory, +  and well-integrated with Google Drive. +  You can share your Colab Notebooks with other people and work collaboratively. +name: Google Colab +title_url: https://colab.research.google.com/notebooks/intro.ipynb +tags: +  topics: +    - general +    - data science +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/hitchhikers_guide_to_python.yaml b/pydis_site/apps/resources/resources/hitchhikers_guide_to_python.yaml new file mode 100644 index 00000000..e48e5717 --- /dev/null +++ b/pydis_site/apps/resources/resources/hitchhikers_guide_to_python.yaml @@ -0,0 +1,18 @@ +description: A best practice handbook for both novice and expert Python developers to the installation, +  configuration, and usage of Python on a daily basis. +name: The Hitchhiker's Guide to Python +title_url: https://python-guide.org/ +urls: +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/28321007-the-hitchhiker-s-guide-to-python +  color: black +tags: +  topics: +    - general +  payment_tiers: +    - paid +  difficulty: +    - beginner +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/reading/books/inferential_thinking.yaml b/pydis_site/apps/resources/resources/inferential_thinking.yaml index 27fad4f7..a8cf2bc8 100644 --- a/pydis_site/apps/resources/resources/reading/books/inferential_thinking.yaml +++ b/pydis_site/apps/resources/resources/inferential_thinking.yaml @@ -2,8 +2,14 @@ description: Inferential Thinking is the textbook for the <a href="http://data8.    It introduces you the fundamentals of both Data Science and Python at a level accessible to all.    It is available both through your browser and in PDF form.  name: Inferential Thinking -position: 13 -urls: -  - icon: regular/link -    url: https://www.inferentialthinking.com/chapters/intro -    color: teal +title_url: https://inferentialthinking.com/chapters/intro +tags: +  topics: +    - data science +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/interactive/_category_info.yaml b/pydis_site/apps/resources/resources/interactive/_category_info.yaml deleted file mode 100644 index 7e8f34d9..00000000 --- a/pydis_site/apps/resources/resources/interactive/_category_info.yaml +++ /dev/null @@ -1,4 +0,0 @@ -description: Learn Python with interactive courses, games, and programming challenges. -name: Interactive -default_icon: branding/python -default_icon_color: black diff --git a/pydis_site/apps/resources/resources/interactive/jetbrains_academy.yaml b/pydis_site/apps/resources/resources/jetbrains_academy.yaml index 937831fa..c3cb7657 100644 --- a/pydis_site/apps/resources/resources/interactive/jetbrains_academy.yaml +++ b/pydis_site/apps/resources/resources/jetbrains_academy.yaml @@ -5,4 +5,14 @@ description: Learn Python with a wide range of high quality, project-based lesso    It requires a paid subscription, but a free trial is available.  name: JetBrains Academy  title_url: https://www.jetbrains.com/academy/ -position: 6 +tags: +  topics: +    - general +    - web development +    - data science +  payment_tiers: +    - subscription +  difficulty: +    - beginner +  type: +    - interactive diff --git a/pydis_site/apps/resources/resources/videos/jetbrains.yaml b/pydis_site/apps/resources/resources/jetbrains_videos.yaml index 5d130db6..00d34e69 100644 --- a/pydis_site/apps/resources/resources/videos/jetbrains.yaml +++ b/pydis_site/apps/resources/resources/jetbrains_videos.yaml @@ -2,11 +2,20 @@ description: A collection of videos made by the PyCharm team at JetBrains on sub    Django, pytest and much more!<br><br>    Episodes of their "What does this package do?" series go over all sorts of libraries in Python    both in the standard library and from the community and give a video explanation of the key concepts. +name: JetBrains YouTube Channel  icon_image: https://upload.wikimedia.org/wikipedia/commons/thumb/1/1a/JetBrains_Logo_2016.svg/1200px-JetBrains_Logo_2016.svg.png  icon_size: 50  title_image: https://resources.jetbrains.com/storage/products/pycharm/img/meta/pycharm_logo_300x300.png -position: 3 -urls: -  - icon: branding/youtube -    url: https://www.youtube.com/channel/UCak6beUTLlVmf0E4AmnQkmw -    color: youtube-red +title_url: https://www.youtube.com/channel/UCak6beUTLlVmf0E4AmnQkmw +tags: +  topics: +    - general +    - testing +    - web development +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - video diff --git a/pydis_site/apps/resources/resources/videos/jim_shaped_coding.yaml b/pydis_site/apps/resources/resources/jim_shaped_coding.yaml index 488cfa83..c9727888 100644 --- a/pydis_site/apps/resources/resources/videos/jim_shaped_coding.yaml +++ b/pydis_site/apps/resources/resources/jim_shaped_coding.yaml @@ -5,9 +5,18 @@ description: 'JimShapedCoding contains a set of YouTube tutorials covering thing      <li><a href="https://www.youtube.com/watch?v=qMrAFscMBBc&list=PLOkVupluCIjvORWaF4kG-sXLgbVemYpEi">Django tutorials</a></li>    </ul>    Check out his channel for more videos!' +name: JimShapedCoding  title_image: https://i.imgur.com/DlovZPf.png -position: 5 -urls: -  - icon: branding/youtube -    url: https://www.youtube.com/channel/UCU8d7rcShA7MGuDyYH1aWGg -    color: youtube-red +title_url: https://www.youtube.com/channel/UCU8d7rcShA7MGuDyYH1aWGg +tags: +  topics: +    - general +    - user interface +    - web development +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - video diff --git a/pydis_site/apps/resources/resources/kaggle_pandas_tutorial.yaml b/pydis_site/apps/resources/resources/kaggle_pandas_tutorial.yaml new file mode 100644 index 00000000..c8e72c6e --- /dev/null +++ b/pydis_site/apps/resources/resources/kaggle_pandas_tutorial.yaml @@ -0,0 +1,13 @@ +description: An interactive tutorial for learning Pandas, the most popular library for manipulating tabular data +  in Python's data science ecosystem. This tutorial assumes some familiarity with writing code in notebooks. +name: Kaggle Pandas Tutorial +title_url: https://www.kaggle.com/learn/pandas +tags: +  topics: +    - data science +  payment_tiers: +    - free +  difficulty: +    - intermediate +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/communities/kivy.yaml b/pydis_site/apps/resources/resources/kivy.yaml index 69f117c7..47ff07ad 100644 --- a/pydis_site/apps/resources/resources/communities/kivy.yaml +++ b/pydis_site/apps/resources/resources/kivy.yaml @@ -5,14 +5,24 @@ icon_image: https://raw.githubusercontent.com/kivy/kivy-website/master/logos/kiv  icon_size: 50  title_image: https://i.imgur.com/EVP3jZR.png  title_url: https://discord.gg/djPtTRJ -position: 6  urls: +  - icon: solid/external-link-alt +    url: https://kivy.org/ +    color: teal    - icon: branding/discord      url: https://discord.gg/djPtTRJ      color: blurple -  - icon: regular/link -    url: https://kivy.org/ -    color: teal    - icon: branding/github      url: https://github.com/kivy      color: black +tags: +  topics: +    - user interface +    - game development +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/communities/microsoft.yaml b/pydis_site/apps/resources/resources/microsoft.yaml index b36c3a85..e1d62955 100644 --- a/pydis_site/apps/resources/resources/communities/microsoft.yaml +++ b/pydis_site/apps/resources/resources/microsoft.yaml @@ -1,12 +1,19 @@  description: Microsoft Python is a Discord server for discussing all things relating to using Python with Microsoft products,    they have channels for Azure, VS Code, IoT, Data Science and much more!  title_image: https://1000logos.net/wp-content/uploads/2017/04/Microsoft-Logo.png -title_url: https://discord.gg/b8YJQPx -position: 1 +title_url: https://www.microsoft.com/en-us/boards/pycon2020.aspx  urls:    - icon: branding/discord      url: https://discord.gg/b8YJQPx      color: blurple -  - icon: regular/link -    url: https://www.microsoft.com/en-us/boards/pycon2020.aspx -    color: teal +tags: +  topics: +    - general +    - tooling +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/videos/microsoft.yaml b/pydis_site/apps/resources/resources/microsoft_videos.yaml index 3ceaa1a2..f45aef63 100644 --- a/pydis_site/apps/resources/resources/videos/microsoft.yaml +++ b/pydis_site/apps/resources/resources/microsoft_videos.yaml @@ -7,12 +7,20 @@ description: A trove of tutorials & guides for developers from Microsoft's Devel    </ul>    Microsoft's Python Development Team also runs a Discord Server for discussions of Python in the Microsoft ecosystem,    including Visual Studio Code and Azure. -title_image: https://img-prod-cms-rt-microsoft-com.akamaized.net/cms/api/am/imageFileData/RE2qVsJ?ver=3f74 -position: 4 +name: Microsoft Developer +title_image: https://upload.wikimedia.org/wikipedia/commons/4/44/Microsoft_logo.svg +title_url: https://www.youtube.com/channel/UCsMica-v34Irf9KVTh6xx-g  urls: -  - icon: branding/youtube -    url: https://www.youtube.com/channel/UCsMica-v34Irf9KVTh6xx-g -    color: youtube-red    - icon: branding/discord      url: https://aka.ms/python-discord      color: blurple +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - video +    - community diff --git a/pydis_site/apps/resources/resources/reading/books/mission_python.yaml b/pydis_site/apps/resources/resources/mission_python.yaml index c4a48b7e..391a2983 100644 --- a/pydis_site/apps/resources/resources/reading/books/mission_python.yaml +++ b/pydis_site/apps/resources/resources/mission_python.yaml @@ -3,11 +3,18 @@ description: Learn programming and Python while building a complete and awesome    images, and walk-throughs make this a pleasure to both read and follow along. Excellent    book for beginners.  name: Mission Python -position: 5 +title_url: https://www.sean.co.uk/books/mission-python/index.shtm  urls: -- icon: regular/link -  url: https://www.sean.co.uk/books/mission-python/index.shtm -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Mission-Python-Code-Space-Adventure/dp/1593278578 -  color: amazon-orange +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/35545850-mission-python +  color: black +tags: +  topics: +    - general +    - game development +  payment_tiers: +    - paid +  difficulty: +    - beginner +  type: +    - book diff --git a/pydis_site/apps/resources/resources/courses/mit_introduction_to_computer_science_and_programming.yaml b/pydis_site/apps/resources/resources/mit_introduction_to_computer_science_and_programming.yaml index 5560b2cb..4e74936d 100644 --- a/pydis_site/apps/resources/resources/courses/mit_introduction_to_computer_science_and_programming.yaml +++ b/pydis_site/apps/resources/resources/mit_introduction_to_computer_science_and_programming.yaml @@ -3,4 +3,14 @@ description: This MITx offering teaches computer science with Python.    and the Python programming language itself.  name: 'MIT: Introduction to Computer Science and Programming'  title_url: https://www.edx.org/course/introduction-computer-science-mitx-6-00-1x-11 -position: 1 +tags: +  topics: +    - general +    - algorithms and data structures +  payment_tiers: +    - free +    - paid +  difficulty: +    - beginner +  type: +    - course diff --git a/pydis_site/apps/resources/resources/tools/editors/mu_editor.yaml b/pydis_site/apps/resources/resources/mu_editor.yaml index b92bac9d..b6318d0e 100644 --- a/pydis_site/apps/resources/resources/tools/editors/mu_editor.yaml +++ b/pydis_site/apps/resources/resources/mu_editor.yaml @@ -4,4 +4,12 @@ description: An editor aimed at beginners for the purpose of learning how to cod    with built-in tools to interact with Adafruit and Arduino boards.  name: Mu-Editor  title_url: https://codewith.mu/ -position: 3 +tags: +  topics: +    - microcontrollers +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/netbats_project_ideas.yaml b/pydis_site/apps/resources/resources/netbats_project_ideas.yaml new file mode 100644 index 00000000..80ba771c --- /dev/null +++ b/pydis_site/apps/resources/resources/netbats_project_ideas.yaml @@ -0,0 +1,14 @@ +description: A repository of project ideas to help one apply what they're learning, maintained by Python +  community member Ned Batchelder, known on Python Discord as nedbat. +name: Ned Batchelder's Kindling Projects +title_url: https://nedbatchelder.com/text/kindling.html +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - project ideas diff --git a/pydis_site/apps/resources/resources/reading/books/neural_networks_from_scratch_in_python.yaml b/pydis_site/apps/resources/resources/neural_networks_from_scratch_in_python.yaml index 974b0e50..c4ad1e1b 100644 --- a/pydis_site/apps/resources/resources/reading/books/neural_networks_from_scratch_in_python.yaml +++ b/pydis_site/apps/resources/resources/neural_networks_from_scratch_in_python.yaml @@ -3,8 +3,17 @@ description: '"Neural Networks From Scratch" is a book intended to teach you how    This is so you can go out and do new/novel things with deep learning as well as to become more successful with even more basic models.    This book is to accompany the usual free tutorial videos and sample code from youtube.com/sentdex.'  name: Neural Networks from Scratch in Python -position: 11 +title_url: https://nnfs.io/  urls: -  - icon: regular/link -    url: https://nnfs.io/ -    color: teal +  - icon: branding/goodreads +    url: https://www.goodreads.com/book/show/55927899-neural-networks-from-scratch-in-python +    color: black +tags: +  topics: +    - data science +  payment_tiers: +    - paid +  difficulty: +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/communities/pallets.yaml b/pydis_site/apps/resources/resources/pallets.yaml index 64c97dab..0da2a625 100644 --- a/pydis_site/apps/resources/resources/communities/pallets.yaml +++ b/pydis_site/apps/resources/resources/pallets.yaml @@ -2,12 +2,18 @@ description: The Pallets Projects develop Python libraries such as the Flask web    the Jinja templating library, and the Click command line toolkit. Join to discuss    and get help from the Pallets community.  title_image: https://i.imgur.com/sV9Ypdf.png -title_url: https://discord.gg/t6rrQZH -position: 7 +title_url: https://www.palletsprojects.com/  urls:    - icon: branding/discord      url: https://discord.gg/t6rrQZH      color: blurple -  - icon: regular/link -    url: https://www.palletsprojects.com/ -    color: teal +tags: +  topics: +    - web development +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/communities/panda3d.yaml b/pydis_site/apps/resources/resources/panda3d.yaml index 47797882..2040450d 100644 --- a/pydis_site/apps/resources/resources/communities/panda3d.yaml +++ b/pydis_site/apps/resources/resources/panda3d.yaml @@ -4,9 +4,20 @@ title_image: https://www.panda3d.org/wp-content/uploads/2019/01/panda3d_logo.png  title_url: https://discord.gg/9XsucTT  position: 9  urls: +  - icon: solid/external-link-alt +    url: https://www.panda3d.org/ +    color: teal    - icon: branding/discord      url: https://discord.gg/9XsucTT      color: blurple -  - icon: regular/link -    url: https://www.panda3d.org/ -    color: teal +tags: +  topics: +    - user interface +    - game development +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/communities/people_postgres_data.yaml b/pydis_site/apps/resources/resources/people_postgres_data.yaml index 1c17d343..46db7095 100644 --- a/pydis_site/apps/resources/resources/communities/people_postgres_data.yaml +++ b/pydis_site/apps/resources/resources/people_postgres_data.yaml @@ -5,14 +5,23 @@ description: People, Postgres, Data specializes in building users of Postgres    and Life in general including movies, games, books and travel.  title_image: https://media.discordapp.net/attachments/748954447857844318/750519488268730377/people_postgres_data.png  title_url: https://discord.gg/Ujw8m8v -position: 2  urls: +  - icon: solid/external-link-alt +    url: https://postgresconf.org/ +    color: teal    - icon: branding/discord      url: https://discord.gg/Ujw8m8v      color: bluple -  - icon: regular/link -    url: https://postgresconf.org/ -    color: teal    - icon: branding/reddit      url: https://reddit.com/r/postgresql      color: orangered +tags: +  topics: +    - databases +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/podcasts/podcast_dunder_init.yaml b/pydis_site/apps/resources/resources/podcast_dunder_init.yaml index efe1601f..2751481a 100644 --- a/pydis_site/apps/resources/resources/podcasts/podcast_dunder_init.yaml +++ b/pydis_site/apps/resources/resources/podcast_dunder_init.yaml @@ -2,4 +2,13 @@ description: The podcast about Python and the people who make it great. Weekly l    interviews with the creators of notable Python packages.  name: Podcast.__init__  title_url: https://www.podcastinit.com/ -position: 2 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - podcast diff --git a/pydis_site/apps/resources/resources/podcasts/_category_info.yaml b/pydis_site/apps/resources/resources/podcasts/_category_info.yaml deleted file mode 100644 index 1d2d3ba5..00000000 --- a/pydis_site/apps/resources/resources/podcasts/_category_info.yaml +++ /dev/null @@ -1,4 +0,0 @@ -description: Notable podcasts about the Python ecosystem. -name: Podcasts -default_icon: regular/microphone-alt -default_icon_color: black diff --git a/pydis_site/apps/resources/resources/courses/practical_python_programming.yaml b/pydis_site/apps/resources/resources/practical_python_programming.yaml index b801ca8c..12873b7c 100644 --- a/pydis_site/apps/resources/resources/courses/practical_python_programming.yaml +++ b/pydis_site/apps/resources/resources/practical_python_programming.yaml @@ -7,3 +7,12 @@ description: Created and taught by <a href="https://dabeaz.com/">David Beazley</  name: Practical Python Programming  title_url: https://dabeaz-course.github.io/practical-python/  position: 4 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - course diff --git a/pydis_site/apps/resources/resources/tools/ides/pycharm.yaml b/pydis_site/apps/resources/resources/pycharm.yaml index b959b0f8..574158bc 100644 --- a/pydis_site/apps/resources/resources/tools/ides/pycharm.yaml +++ b/pydis_site/apps/resources/resources/pycharm.yaml @@ -2,4 +2,13 @@ description: The very best Python IDE, with a wealth of advanced features and co    functions.  name: PyCharm  title_url: https://www.jetbrains.com/pycharm/ -position: 0 +tags: +  topics: +    - general +  payment_tiers: +    - free +    - paid +  difficulty: +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/communities/pyglet.yaml b/pydis_site/apps/resources/resources/pyglet.yaml index 784f514e..a47c7e62 100644 --- a/pydis_site/apps/resources/resources/communities/pyglet.yaml +++ b/pydis_site/apps/resources/resources/pyglet.yaml @@ -4,12 +4,19 @@ description: Pyglet is a powerful,    loading images and videos, and playing sounds and music. All of this with a friendly Pythonic API,    that's simple to learn and doesn't get in your way.  title_image: https://i.imgur.com/LfQwXUe.png -title_url: https://discord.gg/QXyegWe -position: 8 +title_url: http://pyglet.org/  urls:    - icon: branding/discord      url: https://discord.gg/QXyegWe      color: blurple -  - icon: regular/link -    url: http://pyglet.org/ -    color: teal +tags: +  topics: +    - user interface +    - game development +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/podcasts/python_bytes.yaml b/pydis_site/apps/resources/resources/python_bytes.yaml index 4f817f26..9beba4f4 100644 --- a/pydis_site/apps/resources/resources/podcasts/python_bytes.yaml +++ b/pydis_site/apps/resources/resources/python_bytes.yaml @@ -3,3 +3,13 @@ description: A byte-sized podcast where Michael Kennedy and Brian Okken work thr  name: Python Bytes  title_url: https://pythonbytes.fm/  position: 1 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - podcast diff --git a/pydis_site/apps/resources/resources/reading/tutorials/python_cheat_sheet.yaml b/pydis_site/apps/resources/resources/python_cheat_sheet.yaml index 70ac49ef..56f61165 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/python_cheat_sheet.yaml +++ b/pydis_site/apps/resources/resources/python_cheat_sheet.yaml @@ -2,4 +2,12 @@ description: A Python 3 cheat sheet with useful information and tips, as well as    pitfalls for beginners. This is a PDF.  name: Python Cheat Sheet  title_url: https://perso.limsi.fr/pointal/_media/python:cours:mementopython3-english.pdf -position: 6 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/python_cookbook.yaml b/pydis_site/apps/resources/resources/python_cookbook.yaml new file mode 100644 index 00000000..bc05d743 --- /dev/null +++ b/pydis_site/apps/resources/resources/python_cookbook.yaml @@ -0,0 +1,21 @@ +description: A book full of very smart problem-solving recipes for various Python topics, +  including moving from Python 2 to Python 3. +name: Python Cookbook +title_url: http://shop.oreilly.com/product/0636920027072.do +urls: +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/17152735-python-cookbook +  color: black +- icon: branding/github +  url: https://github.com/dabeaz/python-cookbook +  color: black +tags: +  topics: +    - general +    - software design +  payment_tiers: +    - paid +  difficulty: +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/reading/books/python_crash_course.yaml b/pydis_site/apps/resources/resources/python_crash_course.yaml index 3cbf19c8..d916075e 100644 --- a/pydis_site/apps/resources/resources/reading/books/python_crash_course.yaml +++ b/pydis_site/apps/resources/resources/python_crash_course.yaml @@ -7,14 +7,21 @@ description: "This fast-paced, thorough introduction to programming with Python    a Space Invaders–inspired arcade game, a set of data visualizations with Python’s handy libraries,    and a simple web app you can deploy online."  name: Python Crash Course -position: 12 +title_url: https://nostarch.com/pythoncrashcourse2e  urls: -  - icon: regular/link -    url: https://nostarch.com/pythoncrashcourse2e -    color: teal -  - icon: branding/amazon -    url: https://www.amazon.com/Python-Crash-Course-Project-Based-Introduction/dp/1593276036 -    color: amazon-orange +  - icon: branding/goodreads +    url: https://www.goodreads.com/book/show/23241059-python-crash-course +    color: black    - icon: branding/github      url: https://ehmatthes.github.io/pcc/      color: black +tags: +  topics: +    - general +    - game development +  payment_tiers: +    - paid +  difficulty: +    - beginner +  type: +    - book diff --git a/pydis_site/apps/resources/resources/reading/tutorials/python_developer_guide.yaml b/pydis_site/apps/resources/resources/python_developer_guide.yaml index 625d57c8..2806d75d 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/python_developer_guide.yaml +++ b/pydis_site/apps/resources/resources/python_developer_guide.yaml @@ -2,4 +2,12 @@ description: This guide is a comprehensive resource for contributing to Python �    It is maintained by the same community that maintains Python.  name: Python Developer's Guide  title_url: https://devguide.python.org/ -position: 2 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - intermediate +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/python_discord_videos.yaml b/pydis_site/apps/resources/resources/python_discord_videos.yaml new file mode 100644 index 00000000..15a04097 --- /dev/null +++ b/pydis_site/apps/resources/resources/python_discord_videos.yaml @@ -0,0 +1,15 @@ +description: It's our YouTube channel! We are slowly gathering content here directly related to Python, +  our community and the events we host. Come check us out! +title_image: https://raw.githubusercontent.com/python-discord/branding/master/logos/logo_banner/logo_site_banner_dark_512.png +title_url: https://www.youtube.com/pythondiscord +tags: +  topics: +    - general +    - software design +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - video diff --git a/pydis_site/apps/resources/resources/interactive/python_morsels.yaml b/pydis_site/apps/resources/resources/python_morsels.yaml index 879500eb..bbc8133b 100644 --- a/pydis_site/apps/resources/resources/interactive/python_morsels.yaml +++ b/pydis_site/apps/resources/resources/python_morsels.yaml @@ -7,4 +7,13 @@ description: 'Learn to write more idiomatic Python code with deliberate practice    tests and some may include bonuses for a little more of a challenge!'  name: Python Morsels  title_url: https://www.pythonmorsels.com/ -position: 3 +tags: +  topics: +    - general +    - software design +  payment_tiers: +    - subscription +  difficulty: +    - intermediate +  type: +    - interactive diff --git a/pydis_site/apps/resources/resources/communities/subreddit.yaml b/pydis_site/apps/resources/resources/python_subreddit.yaml index d3ddb15a..e94f84fc 100644 --- a/pydis_site/apps/resources/resources/communities/subreddit.yaml +++ b/pydis_site/apps/resources/resources/python_subreddit.yaml @@ -4,3 +4,13 @@ title_icon: branding/reddit  title_icon_color: orangered  title_url: https://www.reddit.com/r/Python/  position: 0 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - community diff --git a/pydis_site/apps/resources/resources/python_tricks.yaml b/pydis_site/apps/resources/resources/python_tricks.yaml new file mode 100644 index 00000000..aa1b2fcd --- /dev/null +++ b/pydis_site/apps/resources/resources/python_tricks.yaml @@ -0,0 +1,19 @@ +description: Full of useful Python tips, tricks and features. Get this if you have +  a good grasp of the basics and want to take your Python skills to the next level, +  or are a experienced programmer looking to add to your toolbelt. +name: Python Tricks +title_url: https://realpython.com/products/python-tricks-book/ +urls: +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/36990732-python-tricks +  color: black +tags: +  topics: +    - general +    - software design +  payment_tiers: +    - paid +  difficulty: +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/interactive/python_tutor.yaml b/pydis_site/apps/resources/resources/python_tutor.yaml index 64b50d09..6bee0d69 100644 --- a/pydis_site/apps/resources/resources/interactive/python_tutor.yaml +++ b/pydis_site/apps/resources/resources/python_tutor.yaml @@ -1,4 +1,14 @@  description: Write Python code in your web browser, and see it visualized step by step.  name: Python Tutor  title_url: https://www.pythontutor.com/ -position: 2 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool +    - interactive diff --git a/pydis_site/apps/resources/resources/reading/_category_info.yaml b/pydis_site/apps/resources/resources/reading/_category_info.yaml deleted file mode 100644 index 64b87e47..00000000 --- a/pydis_site/apps/resources/resources/reading/_category_info.yaml +++ /dev/null @@ -1,2 +0,0 @@ -description: Books and tutorials related to Python and popular third-party libraries and frameworks. -name: Reading diff --git a/pydis_site/apps/resources/resources/reading/books/_category_info.yaml b/pydis_site/apps/resources/resources/reading/books/_category_info.yaml deleted file mode 100644 index ae092a20..00000000 --- a/pydis_site/apps/resources/resources/reading/books/_category_info.yaml +++ /dev/null @@ -1,5 +0,0 @@ -description: The best books for learning Python or Python Frameworks. -name: Books -default_icon: branding/python -default_icon_color: black -position: 0 diff --git a/pydis_site/apps/resources/resources/reading/books/effective_python.yaml b/pydis_site/apps/resources/resources/reading/books/effective_python.yaml deleted file mode 100644 index becd0578..00000000 --- a/pydis_site/apps/resources/resources/reading/books/effective_python.yaml +++ /dev/null @@ -1,15 +0,0 @@ -description: A book that gives 90 best practices for writing excellent Python. Great -  for intermediates. -name: Effective Python -position: 3 -urls: -- icon: regular/link -  url: https://effectivepython.com/ -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Effective-Python-Specific-Software-Development/dp/0134853989 -  color: amazon-orange -  title: Amazon -- icon: branding/github -  url: https://github.com/bslatkin/effectivepython -  color: black diff --git a/pydis_site/apps/resources/resources/reading/books/flask_web_development.yaml b/pydis_site/apps/resources/resources/reading/books/flask_web_development.yaml deleted file mode 100644 index d191f02d..00000000 --- a/pydis_site/apps/resources/resources/reading/books/flask_web_development.yaml +++ /dev/null @@ -1,14 +0,0 @@ -description: A comprehensive Flask walkthrough that has you building a complete social -  blogging application from scratch. -name: Flask Web Development -position: 6 -urls: -- icon: regular/link -  url: https://shop.oreilly.com/product/0636920031116.do -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Flask-Web-Development-Developing-Applications/dp/1449372627 -  color: amazon-orange -- icon: branding/github -  url: https://github.com/miguelgrinberg/flasky -  color: black diff --git a/pydis_site/apps/resources/resources/reading/books/fluent_python.yaml b/pydis_site/apps/resources/resources/reading/books/fluent_python.yaml deleted file mode 100644 index 92f4bbab..00000000 --- a/pydis_site/apps/resources/resources/reading/books/fluent_python.yaml +++ /dev/null @@ -1,14 +0,0 @@ -description: A veritable tome of intermediate and advanced Python information. A must-read -  for any Python professional. By far the most recommended book for intermediates. -name: Fluent Python -position: 7 -urls: -- icon: regular/link -  url: https://www.oreilly.com/library/view/fluent-python/9781491946237/ -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Fluent-Python-Concise-Effective-Programming/dp/1491946008 -  color: amazon-orange -- icon: branding/github -  url: https://github.com/fluentpython -  color: black diff --git a/pydis_site/apps/resources/resources/reading/books/hitchhikers_guide_to_python.yaml b/pydis_site/apps/resources/resources/reading/books/hitchhikers_guide_to_python.yaml deleted file mode 100644 index 906860c7..00000000 --- a/pydis_site/apps/resources/resources/reading/books/hitchhikers_guide_to_python.yaml +++ /dev/null @@ -1,11 +0,0 @@ -description: A best practice handbook for both novice and expert Python developers to the installation, -  configuration, and usage of Python on a daily basis. -name: The Hitchhiker's Guide to Python -position: 0 -urls: -- icon: regular/link -  url: https://python-guide.org/ -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Hitchhikers-Guide-Python-Practices-Development/dp/1491933178/ -  color: amazon-orange diff --git a/pydis_site/apps/resources/resources/reading/books/python_cookbook.yaml b/pydis_site/apps/resources/resources/reading/books/python_cookbook.yaml deleted file mode 100644 index c939ab9e..00000000 --- a/pydis_site/apps/resources/resources/reading/books/python_cookbook.yaml +++ /dev/null @@ -1,14 +0,0 @@ -description: A book full of very smart problem-solving recipes for various Python topics, -  including moving from Python 2 to Python 3. -name: Python Cookbook -position: 8 -urls: -- icon: regular/link -  url: https://shop.oreilly.com/product/0636920027072.do -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Python-Cookbook-Third-David-Beazley/dp/1449340377 -  color: amazon-orange -- icon: branding/github -  url: https://github.com/dabeaz/python-cookbook -  color: black diff --git a/pydis_site/apps/resources/resources/reading/books/python_tricks.yaml b/pydis_site/apps/resources/resources/reading/books/python_tricks.yaml deleted file mode 100644 index c0941809..00000000 --- a/pydis_site/apps/resources/resources/reading/books/python_tricks.yaml +++ /dev/null @@ -1,12 +0,0 @@ -description: Full of useful Python tips, tricks and features. Get this if you have -  a good grasp of the basics and want to take your Python skills to the next level, -  or are a experienced programmer looking to add to your toolbelt. -name: Python Tricks -position: 4 -urls: -- icon: regular/link -  url: https://realpython.com/products/python-tricks-book/ -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Python-Tricks-Buffet-Awesome-Features/dp/1775093301 -  color: amazon-orange diff --git a/pydis_site/apps/resources/resources/reading/books/two_scoops_of_django.yaml b/pydis_site/apps/resources/resources/reading/books/two_scoops_of_django.yaml deleted file mode 100644 index 7d83e7c4..00000000 --- a/pydis_site/apps/resources/resources/reading/books/two_scoops_of_django.yaml +++ /dev/null @@ -1,14 +0,0 @@ -description: Tips, tricks, and best practices for your Django project. -  A highly recommended resource for Django web developers. -name: Two Scoops of Django -position: 9 -urls: -- icon: regular/link -  url: https://twoscoopspress.com/products/two-scoops-of-django-1-11 -  color: teal -- icon: branding/amazon -  url: https://www.amazon.com/Two-Scoops-Django-Best-Practices/dp/0981467342 -  color: amazon-orange -- icon: branding/github -  url: https://github.com/twoscoops/two-scoops-of-django-2.0-code-examples -  color: black diff --git a/pydis_site/apps/resources/resources/reading/tutorials/_category_info.yaml b/pydis_site/apps/resources/resources/reading/tutorials/_category_info.yaml deleted file mode 100644 index a18b837d..00000000 --- a/pydis_site/apps/resources/resources/reading/tutorials/_category_info.yaml +++ /dev/null @@ -1,5 +0,0 @@ -description: Tutorials and references for those that are just getting started with Python. -name: Tutorials -default_icon: branding/python -default_icon_color: black -position: 1 diff --git a/pydis_site/apps/resources/resources/communities/real_python.yaml b/pydis_site/apps/resources/resources/real_python.yaml index 1fc74d93..2ddada03 100644 --- a/pydis_site/apps/resources/resources/communities/real_python.yaml +++ b/pydis_site/apps/resources/resources/real_python.yaml @@ -4,9 +4,18 @@ title_image: https://i.imgur.com/WDqhZ36.png  title_url: https://realpython.com/  position: 3  urls: -  - icon: regular/link -    url: https://realpython.com/ -    color: teal    - icon: branding/youtube      url: https://www.youtube.com/channel/UCI0vQvr9aFn27yR6Ej6n5UA      color: youtube-red +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tutorial +    - video +    - community diff --git a/pydis_site/apps/resources/resources/regex101.yaml b/pydis_site/apps/resources/resources/regex101.yaml new file mode 100644 index 00000000..45d00f1b --- /dev/null +++ b/pydis_site/apps/resources/resources/regex101.yaml @@ -0,0 +1,15 @@ +description: An online tool for testing regular expressions that helps you understand what the regular expression can +  match. Remember to set the "flavor" to Python. +name: regex101 +title_url: https://regex101.com/ +tags: +  topics: +    - general +    - other +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/repl_it.yaml b/pydis_site/apps/resources/resources/repl_it.yaml new file mode 100644 index 00000000..e0f6cbb3 --- /dev/null +++ b/pydis_site/apps/resources/resources/repl_it.yaml @@ -0,0 +1,14 @@ +description: A free, collaborative, in-browser IDE to code in 50+ languages — +  without spending a second on setup. +name: repl.it +title_url: https://repl.it/ +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/tools/accessibility_tools/screen_readers.yaml b/pydis_site/apps/resources/resources/screen_readers.yaml index 39372956..b086b301 100644 --- a/pydis_site/apps/resources/resources/tools/accessibility_tools/screen_readers.yaml +++ b/pydis_site/apps/resources/resources/screen_readers.yaml @@ -4,4 +4,14 @@ description: Screen readers are software programs that allow blind    with this link describing many of them and their capabilities.  name: Screen Readers - American Foundation for the Blind  title_url: https://www.afb.org/blindness-and-low-vision/using-technology/assistive-technology-products/screen-readers -position: 1 +tags: +  topics: +    - other +  payment_tiers: +    - free +    - paid +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/videos/sentdex.yaml b/pydis_site/apps/resources/resources/sentdex.yaml index 4e5f54c6..4f4712ac 100644 --- a/pydis_site/apps/resources/resources/videos/sentdex.yaml +++ b/pydis_site/apps/resources/resources/sentdex.yaml @@ -9,14 +9,24 @@ description: 'An enormous amount of Python content for all skill levels    Check out his channel for more video series!    '  title_image: https://i.imgur.com/kJgWZIu.png -position: 1 +title_url: https://www.youtube.com/user/sentdex  urls: -  - icon: branding/youtube -    url: https://www.youtube.com/user/sentdex -    color: youtube-red +  - icon: solid/external-link-alt +    url: https://pythonprogramming.net/ +    color: teal    - icon: branding/discord      url: https://discord.gg/sentdex      color: blurple -  - icon: regular/link -    url: https://pythonprogramming.net/ -    color: teal +tags: +  topics: +    - general +    - user interface +    - data science +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - video +    - community diff --git a/pydis_site/apps/resources/resources/reading/tutorials/simple_guide_to_git.yaml b/pydis_site/apps/resources/resources/simple_guide_to_git.yaml index 9d151bf9..3bb46e6d 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/simple_guide_to_git.yaml +++ b/pydis_site/apps/resources/resources/simple_guide_to_git.yaml @@ -3,4 +3,12 @@ name: A Simple Guide to Git  title_url: https://rogerdudler.github.io/git-guide/  title_icon: branding/github  title_icon_color: black -position: 4 +tags: +  topics: +    - tooling +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/socratica.yaml b/pydis_site/apps/resources/resources/socratica.yaml new file mode 100644 index 00000000..43d033c0 --- /dev/null +++ b/pydis_site/apps/resources/resources/socratica.yaml @@ -0,0 +1,22 @@ +description: 'Socratica is a small studio focused on producing high quality STEM-related educational content, +including a series about Python. Their videos star actress Ulka Simone Mohanty, who plays an android-like +instructor explaining fundamental concepts in a concise and entertaining way.' +title_image: https://i.imgur.com/4SoHeLz.png +title_url: https://www.youtube.com/playlist?list=PLi01XoE8jYohWFPpC17Z-wWhPOSuh8Er- +urls: +  - icon: solid/database +    url: https://www.youtube.com/playlist?list=PLi01XoE8jYojRqM4qGBF1U90Ee1Ecb5tt +    color: teal +  - icon: branding/youtube +    url: https://www.youtube.com/channel/UCW6TXMZ5Pq6yL6_k5NZ2e0Q +    color: youtube-red +tags: +  topics: +    - general +    - databases +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - video diff --git a/pydis_site/apps/resources/resources/interactive/sololearn.yaml b/pydis_site/apps/resources/resources/sololearn.yaml index 51dceb2a..998f5368 100644 --- a/pydis_site/apps/resources/resources/interactive/sololearn.yaml +++ b/pydis_site/apps/resources/resources/sololearn.yaml @@ -4,4 +4,14 @@ description: SoloLearn's Python 3 course serves as a simple and convenient intro    and mobile apps being available to use.  name: SoloLearn  title_url: https://www.sololearn.com/Course/Python/ -position: 4 +tags: +  topics: +    - general +  payment_tiers: +    - free +    - subscription +  difficulty: +    - beginner +  type: +    - interactive +    - course diff --git a/pydis_site/apps/resources/resources/tools/ides/spyder.yaml b/pydis_site/apps/resources/resources/spyder.yaml index c2f9c2dc..668e9306 100644 --- a/pydis_site/apps/resources/resources/tools/ides/spyder.yaml +++ b/pydis_site/apps/resources/resources/spyder.yaml @@ -2,4 +2,13 @@ description: The Scientific Python Development Environment.    Simpler and lighter than PyCharm, but still packs a punch.  name: Spyder  title_url: https://www.spyder-ide.org/ -position: 1 +tags: +  topics: +    - data science +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/tools/editors/sublime_text.yaml b/pydis_site/apps/resources/resources/sublime_text.yaml index 3c6e7e84..05596477 100644 --- a/pydis_site/apps/resources/resources/tools/editors/sublime_text.yaml +++ b/pydis_site/apps/resources/resources/sublime_text.yaml @@ -2,4 +2,13 @@ description: A powerful Python-backed editor with great community support and a    of extensions.  name: Sublime Text  title_url: https://www.sublimetext.com/ -position: 2 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/podcasts/talk_python_to_me.yaml b/pydis_site/apps/resources/resources/talk_python_to_me.yaml index 5ce21fd7..509922c3 100644 --- a/pydis_site/apps/resources/resources/podcasts/talk_python_to_me.yaml +++ b/pydis_site/apps/resources/resources/talk_python_to_me.yaml @@ -2,4 +2,13 @@ description: The essential weekly Python podcast. Michael Kennedy and a prominen    name within the Python community dive into a topic that relates to their experience.  name: Talk Python To Me  title_url: https://talkpython.fm/ -position: 0 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - podcast diff --git a/pydis_site/apps/resources/resources/tools/accessibility_tools/talon_voice.yaml b/pydis_site/apps/resources/resources/talon_voice.yaml index 9df5f66f..3be5fe20 100644 --- a/pydis_site/apps/resources/resources/tools/accessibility_tools/talon_voice.yaml +++ b/pydis_site/apps/resources/resources/talon_voice.yaml @@ -3,4 +3,13 @@ description: Talon is a tool being built that aims to bring programming,    who have limited or no use of their hands.  name: Talon Voice  title_url: https://talonvoice.com/ -position: 0 +tags: +  topics: +    - other +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/podcasts/test_and_code.yaml b/pydis_site/apps/resources/resources/test_and_code.yaml index d5751577..f0d1c3b3 100644 --- a/pydis_site/apps/resources/resources/podcasts/test_and_code.yaml +++ b/pydis_site/apps/resources/resources/test_and_code.yaml @@ -2,4 +2,14 @@ description: Brian Okken's weekly podcast on testing. Usually deals with Python,    but also covers many language-agnostic topics from the testing and DevOps world.  name: Test & Code  title_url: https://testandcode.com/ -position: 3 +tags: +  topics: +    - testing +    - tooling +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - podcast diff --git a/pydis_site/apps/resources/resources/reading/tutorials/the_flask_mega_tutorial.yaml b/pydis_site/apps/resources/resources/the_flask_mega_tutorial.yaml index 8d61ea73..151768a5 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/the_flask_mega_tutorial.yaml +++ b/pydis_site/apps/resources/resources/the_flask_mega_tutorial.yaml @@ -1,4 +1,13 @@  description: Miguel Grinberg's fully featured mega-tutorial for learning how to create web applications with the Flask framework.  name: The Flask Mega-Tutorial  title_url: https://blog.miguelgrinberg.com/post/the-flask-mega-tutorial-part-i-hello-world -position: 5 +tags: +  topics: +    - web development +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/podcasts/the_real_python_podcast.yaml b/pydis_site/apps/resources/resources/the_real_python_podcast.yaml index dea894ea..647779d5 100644 --- a/pydis_site/apps/resources/resources/podcasts/the_real_python_podcast.yaml +++ b/pydis_site/apps/resources/resources/the_real_python_podcast.yaml @@ -4,4 +4,13 @@ description: A weekly Python podcast hosted by Christopher Bailey with interview    career tips, and related software development topics.  name: The Real Python Podcast  title_url: https://realpython.com/podcasts/rpp/ -position: 4 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - podcast diff --git a/pydis_site/apps/resources/resources/reading/books/think_python.yaml b/pydis_site/apps/resources/resources/think_python.yaml index 6de87043..7099afd8 100644 --- a/pydis_site/apps/resources/resources/reading/books/think_python.yaml +++ b/pydis_site/apps/resources/resources/think_python.yaml @@ -4,14 +4,21 @@ description: Think Python is an introduction to Python programming for beginners    Larger pieces, like recursion and object-oriented programming are divided into a sequence of smaller steps    and introduced over the course of several chapters.  name: Think Python -position: 10 +title_url: https://greenteapress.com/wp/think-python-2e/  urls: -  - icon: regular/link -    url: https://greenteapress.com/wp/think-python-2e/ -    color: teal -  - icon: branding/amazon -    url: https://www.amazon.com/gp/product/1491939362 -    color: amazon-orange +  - icon: branding/goodreads +    url: https://www.goodreads.com/book/show/14514306-think-python +    color: black    - icon: branding/github      url: https://github.com/AllenDowney/ThinkPython2      color: black +tags: +  topics: +    - general +    - software design +  payment_tiers: +    - paid +  difficulty: +    - beginner +  type: +    - book diff --git a/pydis_site/apps/resources/resources/tools/ides/thonny.yaml b/pydis_site/apps/resources/resources/thonny.yaml index d7f03a74..29ba9e07 100644 --- a/pydis_site/apps/resources/resources/tools/ides/thonny.yaml +++ b/pydis_site/apps/resources/resources/thonny.yaml @@ -3,3 +3,12 @@ description: A Python IDE specifically aimed at learning programming. Has a lot  name: Thonny  title_url: https://thonny.org/  position: 2 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/tools/_category_info.yaml b/pydis_site/apps/resources/resources/tools/_category_info.yaml deleted file mode 100644 index 6b16baa6..00000000 --- a/pydis_site/apps/resources/resources/tools/_category_info.yaml +++ /dev/null @@ -1,4 +0,0 @@ -description: This page is a curated list of tools that we regularly recommend in the community. -  If you have a suggestion for something to add to this page, please create an issue in -  <a href="https://github.com/python-discord/meta/issues">our meta repo</a>, and we'll consider adding it. -name: Tools diff --git a/pydis_site/apps/resources/resources/tools/accessibility_tools/_category_info.yaml b/pydis_site/apps/resources/resources/tools/accessibility_tools/_category_info.yaml deleted file mode 100644 index e770db07..00000000 --- a/pydis_site/apps/resources/resources/tools/accessibility_tools/_category_info.yaml +++ /dev/null @@ -1,5 +0,0 @@ -description: Accessibility tools that help people write Python code. -name: Accessibility Tools -default_icon: branding/python -default_icon_color: black -position: 2 diff --git a/pydis_site/apps/resources/resources/tools/editors/_category_info.yaml b/pydis_site/apps/resources/resources/tools/editors/_category_info.yaml deleted file mode 100644 index 3cdfff3a..00000000 --- a/pydis_site/apps/resources/resources/tools/editors/_category_info.yaml +++ /dev/null @@ -1,5 +0,0 @@ -description: Lightweight code editors supporting Python -name: Editors -default_icon: branding/python -default_icon_color: black -position: 1 diff --git a/pydis_site/apps/resources/resources/tools/editors/google_collab.yaml b/pydis_site/apps/resources/resources/tools/editors/google_collab.yaml deleted file mode 100644 index 302c3e2e..00000000 --- a/pydis_site/apps/resources/resources/tools/editors/google_collab.yaml +++ /dev/null @@ -1,7 +0,0 @@ -description: Google Collab is a high-powered custom version of Jupyter Notebook which supports e.g. -  !apt-get to install arbitrary Debian packages to the runtime, which is very generous with CPU and memory, -  and well-integrated with Google Drive. -  You can share your Collab Notebooks with other people and work collaboratively. -name: Google Collab -title_url: https://colab.research.google.com/notebooks/intro.ipynb -position: 4 diff --git a/pydis_site/apps/resources/resources/tools/ides/_category_info.yaml b/pydis_site/apps/resources/resources/tools/ides/_category_info.yaml deleted file mode 100644 index 614625a6..00000000 --- a/pydis_site/apps/resources/resources/tools/ides/_category_info.yaml +++ /dev/null @@ -1,5 +0,0 @@ -description: Fully-integrated development environments for serious Python work. -name: IDEs -default_icon: branding/python -default_icon_color: black -position: 0 diff --git a/pydis_site/apps/resources/resources/tools/ides/replit.yaml b/pydis_site/apps/resources/resources/tools/ides/replit.yaml deleted file mode 100644 index 844c5016..00000000 --- a/pydis_site/apps/resources/resources/tools/ides/replit.yaml +++ /dev/null @@ -1,5 +0,0 @@ -description: A free, collaborative, in-browser IDE to code in 50+ languages — -  without spending a second on setup. -name: replit -title_url: https://replit.com/ -position: 3 diff --git a/pydis_site/apps/resources/resources/two_scoops_of_django.yaml b/pydis_site/apps/resources/resources/two_scoops_of_django.yaml new file mode 100644 index 00000000..96eafd28 --- /dev/null +++ b/pydis_site/apps/resources/resources/two_scoops_of_django.yaml @@ -0,0 +1,20 @@ +description: Tips, tricks, and best practices for your Django project. +  A highly recommended resource for Django web developers. +name: Two Scoops of Django +title_url: https://www.feldroy.com/collections/everything/products/two-scoops-of-django-3-x +urls: +- icon: branding/goodreads +  url: https://www.goodreads.com/book/show/55822151-two-scoops-of-django-3-x +  color: black +- icon: branding/github +  url: https://github.com/twoscoops/two-scoops-of-django-2.0-code-examples +  color: black +tags: +  topics: +    - web development +  payment_tiers: +    - paid +  difficulty: +    - intermediate +  type: +    - book diff --git a/pydis_site/apps/resources/resources/courses/university_of_michigan.yaml b/pydis_site/apps/resources/resources/university_of_michigan.yaml index 3efe7640..7aaaf2ae 100644 --- a/pydis_site/apps/resources/resources/courses/university_of_michigan.yaml +++ b/pydis_site/apps/resources/resources/university_of_michigan.yaml @@ -2,4 +2,12 @@ description: A 5-part specialization course that teaches Python from scratch.    The course has no pre-requisites and avoids all but the simplest mathematics.  name: 'University of Michigan: Programming for Everybody'  title_url: https://www.coursera.org/learn/python -position: 2 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +  type: +    - course diff --git a/pydis_site/apps/resources/resources/courses/university_of_toronto.yaml b/pydis_site/apps/resources/resources/university_of_toronto.yaml index 0a7839de..94df96f2 100644 --- a/pydis_site/apps/resources/resources/courses/university_of_toronto.yaml +++ b/pydis_site/apps/resources/resources/university_of_toronto.yaml @@ -1,7 +1,6 @@  description: A 2-part course that teaches Python. Primarily intended for high school students    and first-year university students who want to learn programming.  name: 'University of Toronto: Learn to Program' -position: 0  urls:    - icon: regular/graduation-cap      url: https://www.coursera.org/learn/learn-to-program @@ -9,3 +8,13 @@ urls:    - icon: regular/graduation-cap      url: https://www.coursera.org/learn/program-code      color: youtube-red +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - course diff --git a/pydis_site/apps/resources/resources/vcokltfre_discord_bot_tutorial.yaml b/pydis_site/apps/resources/resources/vcokltfre_discord_bot_tutorial.yaml new file mode 100644 index 00000000..32476dab --- /dev/null +++ b/pydis_site/apps/resources/resources/vcokltfre_discord_bot_tutorial.yaml @@ -0,0 +1,14 @@ +description: This tutorial, written by Python Discord staff member vcokltfre, +  will walk you through all the aspects of creating your own Discord bot, +  starting from from creating the bot user itself. +name: vcokltfre's Discord Bot Tutorial +title_url: https://tutorial.vcokltfre.dev/ +tags: +  topics: +    - discord bots +  payment_tiers: +    - free +  difficulty: +    - intermediate +  type: +    - tutorial diff --git a/pydis_site/apps/resources/resources/videos/_category_info.yaml b/pydis_site/apps/resources/resources/videos/_category_info.yaml deleted file mode 100644 index 8192e021..00000000 --- a/pydis_site/apps/resources/resources/videos/_category_info.yaml +++ /dev/null @@ -1,2 +0,0 @@ -description: Excellent Youtube channels with content related to Python. -name: Videos diff --git a/pydis_site/apps/resources/resources/videos/python_discord.yaml b/pydis_site/apps/resources/resources/videos/python_discord.yaml deleted file mode 100644 index 04235b08..00000000 --- a/pydis_site/apps/resources/resources/videos/python_discord.yaml +++ /dev/null @@ -1,8 +0,0 @@ -description: It's our channel! We are slowly gathering content here directly related to Python, -  our community and the events we host. Come check us out! -title_image: https://raw.githubusercontent.com/python-discord/branding/master/logos/logo_banner/logo_site_banner_dark_512.png -position: 2 -urls: -  - icon: branding/youtube -    url: https://www.youtube.com/pythondiscord -    color: youtube-red diff --git a/pydis_site/apps/resources/resources/tools/editors/visual_studio_code.yaml b/pydis_site/apps/resources/resources/visual_studio_code.yaml index e3737ca7..3cf858f8 100644 --- a/pydis_site/apps/resources/resources/tools/editors/visual_studio_code.yaml +++ b/pydis_site/apps/resources/resources/visual_studio_code.yaml @@ -1,4 +1,13 @@  description: A fully-featured editor based on Electron, extendable with plugins.  name: Visual Studio Code  title_url: https://code.visualstudio.com/ -position: 1 +tags: +  topics: +    - general +  payment_tiers: +    - free +  difficulty: +    - beginner +    - intermediate +  type: +    - tool diff --git a/pydis_site/apps/resources/resources/reading/tutorials/wtf_python.yaml b/pydis_site/apps/resources/resources/wtf_python.yaml index a25a84fd..6d90ba39 100644 --- a/pydis_site/apps/resources/resources/reading/tutorials/wtf_python.yaml +++ b/pydis_site/apps/resources/resources/wtf_python.yaml @@ -6,3 +6,13 @@ description: Python, being a beautifully designed high-level and interpreter-bas  name: WTF Python  title_url: https://github.com/satwikkansal/wtfpython  position: 7 +tags: +  topics: +    - software design +    - other +  payment_tiers: +    - free +  difficulty: +    - intermediate +  type: +    - tutorial diff --git a/pydis_site/apps/resources/templatetags/get_category_icon.py b/pydis_site/apps/resources/templatetags/get_category_icon.py new file mode 100644 index 00000000..71f1393f --- /dev/null +++ b/pydis_site/apps/resources/templatetags/get_category_icon.py @@ -0,0 +1,39 @@ +from django import template + +register = template.Library() + +_ICONS = { +    "Algorithms And Data Structures": "fa-cogs", +    "Beginner": "fa-play-circle", +    "Book": "fa-book", +    "Community": "fa-users", +    "Course": "fa-chalkboard-teacher", +    "Data Science": "fa-flask", +    "Databases": "fa-server", +    "Discord Bots": "fa-robot", +    "Free": "fa-first-aid", +    "Game Development": "fa-gamepad", +    "General": "fa-book", +    "Interactive": "fa-mouse-pointer", +    "Intermediate": "fa-align-center", +    "Microcontrollers": "fa-microchip", +    "Other": "fa-question-circle", +    "Paid": "fa-dollar-sign", +    "Podcast": "fa-microphone-alt", +    "Project Ideas": "fa-lightbulb-o", +    "Software Design": "fa-paint-brush", +    "Subscription": "fa-credit-card", +    "Testing": "fa-vial", +    "Tool": "fa-tools", +    "Tooling": "fa-toolbox", +    "Tutorial": "fa-clipboard-list", +    "User Interface": "fa-desktop", +    "Video": "fa-video", +    "Web Development": "fa-wifi", +} + + +def get_category_icon(name: str) -> str: +    """Get icon of a specific resource category.""" +    return f'fa {_ICONS[name]}' diff --git a/pydis_site/apps/resources/templatetags/to_kebabcase.py b/pydis_site/apps/resources/templatetags/to_kebabcase.py new file mode 100644 index 00000000..41e2ac85 --- /dev/null +++ b/pydis_site/apps/resources/templatetags/to_kebabcase.py @@ -0,0 +1,39 @@ +import re + +from django import template + +REGEX_CONSECUTIVE_NON_LETTERS = r"[^A-Za-z0-9]+" +register = template.Library() + + +def _to_kebabcase(class_name: str) -> str: +    """ +    Convert any string to kebab-case. + +    For example, convert +    "__Favorite FROOT¤#/$?is----LeMON???" to +    "favorite-froot-is-lemon" +    """ +    # First, make it lowercase, and just remove any apostrophes. +    # We remove the apostrophes because "wasnt" is better than "wasn-t" +    class_name = class_name.casefold() +    class_name = class_name.replace("'", '') + +    # Now, replace any non-letter that remains with a dash. +    # If there are multiple consecutive non-letters, just replace them with a single dash. +    # my-favorite-class is better than my-favorite------class +    class_name = re.sub( +        REGEX_CONSECUTIVE_NON_LETTERS, +        "-", +        class_name, +    ) + +    # Now we use strip to get rid of any leading or trailing dashes. +    class_name = class_name.strip("-") +    return class_name + + +def to_kebabcase(class_name: str) -> str: +    """Convert a string to kebab-case.""" +    return _to_kebabcase(class_name) diff --git a/pydis_site/apps/resources/tests/test_to_kebabcase.py b/pydis_site/apps/resources/tests/test_to_kebabcase.py new file mode 100644 index 00000000..a141143d --- /dev/null +++ b/pydis_site/apps/resources/tests/test_to_kebabcase.py @@ -0,0 +1,19 @@ +from django.test import TestCase + +from pydis_site.apps.resources.templatetags.to_kebabcase import _to_kebabcase + + +class TestToKebabcase(TestCase): +    """Tests for the `as_css_class` template tag.""" + +    def test_to_kebabcase(self): +        """Test the to_kebabcase utility and template tag.""" +        weird_input = ( +            "_-_--_A_LEm0n?in&¤'the##trEE£$@€@€@@£is-NOT----QUITE//" +            "as#good!  as one   __IN-YOUR|||HaND" +        ) + +        self.assertEqual( +            _to_kebabcase(weird_input), +            "a-lem0n-in-the-tree-is-not-quite-as-good-as-one-in-your-hand", +        ) diff --git a/pydis_site/apps/resources/tests/test_views.py b/pydis_site/apps/resources/tests/test_views.py index 3ad0b958..a2a203ce 100644 --- a/pydis_site/apps/resources/tests/test_views.py +++ b/pydis_site/apps/resources/tests/test_views.py @@ -1,5 +1,4 @@  from pathlib import Path -from unittest.mock import patch  from django.conf import settings  from django.test import TestCase @@ -17,18 +16,14 @@ class TestResourcesView(TestCase):          response = self.client.get(url)          self.assertEqual(response.status_code, 200) - -class TestResourcesListView(TestCase): -    @patch("pydis_site.apps.resources.views.resources_list.RESOURCES_PATH", TESTING_RESOURCES_PATH) -    def test_valid_resource_list_200(self): -        """Check does site return code 200 when visiting valid resource list.""" -        url = reverse("resources:resources", args=("testing",)) +    def test_resources_with_valid_argument(self): +        """Check that you can resolve the resources when passing a valid argument.""" +        url = reverse("resources:index", kwargs={"resource_type": "book"})          response = self.client.get(url)          self.assertEqual(response.status_code, 200) -    @patch("pydis_site.apps.resources.views.resources_list.RESOURCES_PATH", TESTING_RESOURCES_PATH) -    def test_invalid_resource_list_404(self): -        """Check does site return code 404 when trying to visit invalid resource list.""" -        url = reverse("resources:resources", args=("invalid",)) +    def test_resources_with_invalid_argument(self): +        """Check that you can resolve the resources when passing an invalid argument.""" +        url = reverse("resources:index", kwargs={"resource_type": "urinal-cake"})          response = self.client.get(url)          self.assertEqual(response.status_code, 404) diff --git a/pydis_site/apps/resources/tests/testing_resources/testing/_category_info.yaml b/pydis_site/apps/resources/tests/testing_resources/testing/_category_info.yaml deleted file mode 100644 index bae17ea3..00000000 --- a/pydis_site/apps/resources/tests/testing_resources/testing/_category_info.yaml +++ /dev/null @@ -1 +0,0 @@ -name: Testing diff --git a/pydis_site/apps/resources/tests/testing_resources/testing/foobar/_category_info.yaml b/pydis_site/apps/resources/tests/testing_resources/testing/foobar/_category_info.yaml deleted file mode 100644 index eaac32d9..00000000 --- a/pydis_site/apps/resources/tests/testing_resources/testing/foobar/_category_info.yaml +++ /dev/null @@ -1 +0,0 @@ -name: Foobar diff --git a/pydis_site/apps/resources/tests/testing_resources/testing/foobar/resource_test.yaml b/pydis_site/apps/resources/tests/testing_resources/testing/foobar/resource_test.yaml deleted file mode 100644 index 22835090..00000000 --- a/pydis_site/apps/resources/tests/testing_resources/testing/foobar/resource_test.yaml +++ /dev/null @@ -1 +0,0 @@ -name: Resource Test diff --git a/pydis_site/apps/resources/tests/testing_resources/testing/my_resource.yaml b/pydis_site/apps/resources/tests/testing_resources/testing/my_resource.yaml deleted file mode 100644 index 61df6173..00000000 --- a/pydis_site/apps/resources/tests/testing_resources/testing/my_resource.yaml +++ /dev/null @@ -1 +0,0 @@ -name: My Resource diff --git a/pydis_site/apps/resources/urls.py b/pydis_site/apps/resources/urls.py index 10eda132..ed24dc99 100644 --- a/pydis_site/apps/resources/urls.py +++ b/pydis_site/apps/resources/urls.py @@ -1,25 +1,9 @@ -import typing -from pathlib import Path -  from django_distill import distill_path  from pydis_site.apps.resources import views  app_name = "resources" - - -def get_all_resources() -> typing.Iterator[dict[str, str]]: -    """Yield a dict of all resource categories.""" -    for category in Path("pydis_site", "apps", "resources", "resources").iterdir(): -        yield {"category": category.name} - -  urlpatterns = [ -    distill_path("", views.ResourcesView.as_view(), name="index"), -    distill_path( -        "<str:category>/", -        views.ResourcesListView.as_view(), -        name="resources", -        distill_func=get_all_resources -    ), +    distill_path("", views.resources.ResourceView.as_view(), name="index"), +    distill_path("<resource_type>/", views.resources.ResourceView.as_view(), name="index"),  ] diff --git a/pydis_site/apps/resources/utils.py b/pydis_site/apps/resources/utils.py deleted file mode 100644 index 1855fc80..00000000 --- a/pydis_site/apps/resources/utils.py +++ /dev/null @@ -1,42 +0,0 @@ -import typing as t -from pathlib import Path - -import yaml - - -def get_resources(path: Path) -> t.List[t.Dict]: -    """Loads resource YAMLs from provided path.""" -    resources = [] - -    for item in path.iterdir(): -        if item.is_file() and item.suffix == ".yaml" and item.name != "_category_info.yaml": -            resources.append(yaml.safe_load(item.read_text())) - -    return resources - - -def get_subcategories(path: Path) -> t.List[t.Dict]: -    """Loads resources subcategories with their resources by provided path.""" -    subcategories = [] - -    for item in path.iterdir(): -        if item.is_dir() and item.joinpath("_category_info.yaml").exists(): -            subcategories.append({ -                "category_info": { -                    **yaml.safe_load( -                        item.joinpath("_category_info.yaml").read_text() -                    ), -                    "raw_name": item.name -                }, -                "resources": [ -                    yaml.safe_load(subitem.read_text()) -                    for subitem in item.iterdir() -                    if ( -                        subitem.is_file() -                        and subitem.suffix == ".yaml" -                        and subitem.name != "_category_info.yaml" -                    ) -                ] -            }) - -    return subcategories diff --git a/pydis_site/apps/resources/views/__init__.py b/pydis_site/apps/resources/views/__init__.py index 8eb383b5..986f3e10 100644 --- a/pydis_site/apps/resources/views/__init__.py +++ b/pydis_site/apps/resources/views/__init__.py @@ -1,4 +1,3 @@ -from .resources import ResourcesView -from .resources_list import ResourcesListView +from .resources import ResourceView -__all__ = ["ResourcesView", "ResourcesListView"] +__all__ = ["ResourceView"] diff --git a/pydis_site/apps/resources/views/resources.py b/pydis_site/apps/resources/views/resources.py index 25ce3e50..2375f722 100644 --- a/pydis_site/apps/resources/views/resources.py +++ b/pydis_site/apps/resources/views/resources.py @@ -1,7 +1,126 @@ -from django.views.generic import TemplateView +import json +import typing as t +from pathlib import Path +import yaml +from django.core.handlers.wsgi import WSGIRequest +from django.http import HttpResponse, HttpResponseNotFound +from django.shortcuts import render +from django.views import View -class ResourcesView(TemplateView): -    """View for resources index page.""" +from pydis_site import settings +from pydis_site.apps.resources.templatetags.to_kebabcase import to_kebabcase -    template_name = "resources/resources.html" +RESOURCES_PATH = Path(settings.BASE_DIR, "pydis_site", "apps", "resources", "resources") + + +class ResourceView(View): +    """Our curated list of good learning resources.""" + +    @staticmethod +    def _sort_key_disregard_the(tuple_: tuple) -> str: +        """Sort a tuple by its key alphabetically, disregarding 'the' as a prefix.""" +        name, resource = tuple_ +        name = name.casefold() +        if name.startswith("the ") or name.startswith("the_"): +            return name[4:] +        return name + +    def __init__(self, *args, **kwargs): +        """Set up all the resources.""" +        super().__init__(*args, **kwargs) + +        # Load the resources from the yaml files in /resources/ +        self.resources = { +            path.stem: yaml.safe_load(path.read_text()) +            for path in RESOURCES_PATH.rglob("*.yaml") +        } + +        # Sort the resources alphabetically +        self.resources = dict(sorted(self.resources.items(), key=self._sort_key_disregard_the)) + +        # Parse out all current tags +        resource_tags = { +            "topics": set(), +            "payment_tiers": set(), +            "difficulty": set(), +            "type": set(), +        } +        for resource_name, resource in self.resources.items(): +            css_classes = [] +            for tag_type in resource_tags.keys(): +                # Store the tags into `resource_tags` +                tags = resource.get("tags", {}).get(tag_type, []) +                for tag in tags: +                    tag = tag.title() +                    tag = tag.replace("And", "and") +                    resource_tags[tag_type].add(tag) + +                # Make a CSS class friendly representation too, while we're already iterating. +                for tag in tags: +                    css_tag = to_kebabcase(f"{tag_type}-{tag}") +                    css_classes.append(css_tag) + +            # Now add the css classes back to the resource, so we can use them in the template. +            self.resources[resource_name]["css_classes"] = " ".join(css_classes) + +        # Set up all the filter checkbox metadata +        self.filters = { +            "Difficulty": { +                "filters": sorted(resource_tags.get("difficulty")), +                "icon": "fas fa-brain", +                "hidden": False, +            }, +            "Type": { +                "filters": sorted(resource_tags.get("type")), +                "icon": "fas fa-photo-video", +                "hidden": False, +            }, +            "Payment tiers": { +                "filters": sorted(resource_tags.get("payment_tiers")), +                "icon": "fas fa-dollar-sign", +                "hidden": True, +            }, +            "Topics": { +                "filters": sorted(resource_tags.get("topics")), +                "icon": "fas fa-lightbulb", +                "hidden": True, +            } +        } + +        # The bottom topic should always be "Other". +        self.filters["Topics"]["filters"].remove("Other") +        self.filters["Topics"]["filters"].append("Other") + +        # A complete list of valid filter names +        self.valid_filters = { +            "topics": [to_kebabcase(topic) for topic in self.filters["Topics"]["filters"]], +            "payment_tiers": [ +                to_kebabcase(tier) for tier in self.filters["Payment tiers"]["filters"] +            ], +            "type": [to_kebabcase(type_) for type_ in self.filters["Type"]["filters"]], +            "difficulty": [to_kebabcase(tier) for tier in self.filters["Difficulty"]["filters"]], +        } + +    def get(self, request: WSGIRequest, resource_type: t.Optional[str] = None) -> HttpResponse: +        """List out all the resources, and any filtering options from the URL.""" +        # Add type filtering if the request is made to somewhere like /resources/video. +        # We also convert all spaces to dashes, so they'll correspond with the filters. +        if resource_type: +            dashless_resource_type = resource_type.replace("-", " ") + +            if dashless_resource_type.title() not in self.filters["Type"]["filters"]: +                return HttpResponseNotFound() + +            resource_type = resource_type.replace(" ", "-") + +        return render( +            request, +            template_name="resources/resources.html", +            context={ +                "resources": self.resources, +                "filters": self.filters, +                "valid_filters": json.dumps(self.valid_filters), +                "resource_type": resource_type, +            } +        ) diff --git a/pydis_site/apps/resources/views/resources_list.py b/pydis_site/apps/resources/views/resources_list.py deleted file mode 100644 index 55f22993..00000000 --- a/pydis_site/apps/resources/views/resources_list.py +++ /dev/null @@ -1,39 +0,0 @@ -from pathlib import Path -from typing import Any, Dict - -import yaml -from django.conf import settings -from django.http import Http404 -from django.views.generic import TemplateView - -from pydis_site.apps.resources.utils import get_resources, get_subcategories - -RESOURCES_PATH = Path(settings.BASE_DIR, "pydis_site", "apps", "resources", "resources") - - -class ResourcesListView(TemplateView): -    """Shows specific resources list.""" - -    template_name = "resources/resources_list.html" - -    def get_context_data(self, **kwargs) -> Dict[str, Any]: -        """Add resources and subcategories data into context.""" -        context = super().get_context_data(**kwargs) - -        resource_path = RESOURCES_PATH / self.kwargs["category"] -        if ( -                not resource_path.is_dir() -                or not resource_path.joinpath("_category_info.yaml").exists() -        ): -            raise Http404 - -        context["resources"] = get_resources(resource_path) -        context["subcategories"] = get_subcategories(resource_path) -        context["category_info"] = { -            **yaml.safe_load( -                resource_path.joinpath("_category_info.yaml").read_text() -            ), -            "raw_name": resource_path.name -        } - -        return context diff --git a/pydis_site/settings.py b/pydis_site/settings.py index d38c298b..3b146f2c 100644 --- a/pydis_site/settings.py +++ b/pydis_site/settings.py @@ -272,6 +272,7 @@ BULMA_SETTINGS = {          "bulma-dropdown",          "bulma-navbar-burger",      ], +    "fontawesome_token": "ff22cb6f41",  }  # Information about site repository diff --git a/pydis_site/static/css/collapsibles.css b/pydis_site/static/css/collapsibles.css new file mode 100644 index 00000000..1d73fa00 --- /dev/null +++ b/pydis_site/static/css/collapsibles.css @@ -0,0 +1,11 @@ +.collapsible { +  cursor: pointer; +  width: 100%; +  border: none; +  outline: none; +} + +.collapsible-content { +  transition: max-height 0.3s ease-out; +  overflow: hidden; +} diff --git a/pydis_site/static/css/content/page.css b/pydis_site/static/css/content/page.css index 2d4bd325..d831f86d 100644 --- a/pydis_site/static/css/content/page.css +++ b/pydis_site/static/css/content/page.css @@ -77,16 +77,3 @@ ul.menu-list.toc {  li img {      margin-top: 0.5em;  } - -.collapsible { -  cursor: pointer; -  width: 100%; -  border: none; -  outline: none; -} - -.collapsible-content { -  overflow: hidden; -  max-height: 0; -  transition: max-height 0.2s ease-out; -} diff --git a/pydis_site/static/css/resources/resources.css b/pydis_site/static/css/resources/resources.css index cf4cb472..b8456e38 100644 --- a/pydis_site/static/css/resources/resources.css +++ b/pydis_site/static/css/resources/resources.css @@ -1,29 +1,256 @@ -.box, .tile.is-parent { -    transition: 0.1s ease-out; +/* Colors for icons */ +i.resource-icon.is-orangered { +    color: #FE640A;  } -.box { -    min-height: 15vh; +i.resource-icon.is-blurple { +    color: #7289DA;  } -.tile.is-parent:hover .box { -    box-shadow: 0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23); +i.resource-icon.is-teal { +    color: #95DBE5;  } -.tile.is-parent:hover { -    padding: 0.65rem 0.85rem 0.85rem 0.65rem; -    filter: saturate(1.1) brightness(1.1); +i.resource-icon.is-youtube-red { +    color: #BB0000; +} +i.resource-icon.is-black { +    color: #2c3334; +} + +/* Colors when icons are hovered */ +i.resource-icon.is-hoverable:hover { +    filter: brightness(125%); +} +i.resource-icon.is-hoverable.is-black:hover { +    filter: brightness(170%); +} +i.resource-icon.is-hoverable.is-teal:hover { +    filter: brightness(80%); +} + +/* Icon padding */ +.breadcrumb-section { +    padding: 1rem; +} +i.has-icon-padding { +    padding: 0 10px 25px 0; +} +#tab-content p { +    display: none; +} +#tab-content p.is-active { +display: block; +} + +/* Disable highlighting for all text in the filters. */ +.filter-checkbox, +.filter-panel label, +.card-header span { +    user-select: none +} + +/* Remove pointless margin in panel header */ +#filter-panel-header { +    margin-bottom: 0; +} + +/* Full width filter cards */ +#resource-filtering-panel .card .collapsible-content .card-content { +    padding:0 +} + +/* Don't round the corners of the collapsibles */ +.filter-category-header { +    border-radius: 0;  } -#readingBlock { -    background-image: linear-gradient(141deg, #911eb4 0%, #b631de 71%, #cf4bf7 100%); +/* Make the checkboxes indent under the filter headers */ +.filter-category-header .card-header .card-header-title { +    padding-left: 0; +} +.filter-panel { +    padding-left: 1.5rem; +} +.filter-checkbox { +    margin-right: 0.25em !important; +} + +/* Center the 404 div */ +.no-resources-found { +    display: none; +    flex-direction: column; +    align-items: center; +    margin-top: 1em; +} + +/* Make sure jQuery will use flex when setting `show()` again. */ +.no-resources-found[style*='display: block'] { +    display: flex !important; +} + +/* Disable clicking on the checkbox itself. */ +/* Instead, we want to let the anchor tag handle clicks. */ +.filter-checkbox { +    pointer-events: none; +} + +/* Blurple category icons */ +i.is-primary { +    color: #7289da; +} + +/* A little space above the filter card, please! */ +.filter-tags { +    padding-bottom: .5em; +} + +/* Style the close all filters button */ +.close-filters-button { +    margin-left: auto; +    display:none; +} +.close-filters-button a { +    height: fit-content; +    width: fit-content; +    margin-right: 6px; +} +.close-filters-button a i { +    color: #939bb3; +} +.close-filters-button a i:hover { +    filter: brightness(115%);  } -#interactiveBlock { -    background-image: linear-gradient(141deg, #d05600 0%, #da722a 71%, #e68846 100%); +/* When hovering title anchors, just make the color a lighter primary, not black. */ +.resource-box a:hover { +    filter: brightness(120%); +    color: #7289DA; +} + + +/* Set default display to inline-flex, for centering. */ +span.filter-box-tag { +    display: none; +    align-items: center; +    cursor: pointer; +    user-select: none; +} + +/* Make sure jQuery will use inline-flex when setting `show()` again. */ +span.filter-box-tag[style*='display: block'] { +    display: inline-flex !important; +} + +/* Make resource tags clickable */ +.resource-tag { +    cursor: pointer; +    user-select: none; +} + +/* Give the resource tags a bit of breathing room */ +.resource-tag-container { +    padding-left: 1.5rem; +} + +/* When hovering tags, brighten them a bit. */ +.resource-tag:hover, +.filter-box-tag:hover { +    filter: brightness(95%); +} + +/* Move the x down 1 pixel to align center */ +button.delete { +    margin-top: 1px; +} + +/* Colors for delete button x's */ +button.delete.is-primary::before, +button.delete.is-primary::after { +    background-color: #2a45a2; +} +button.delete.is-success::before, +button.delete.is-success::after { +    background-color: #2c9659; +} +button.delete.is-danger::before, +button.delete.is-danger::after { +    background-color: #c32841; +} +button.delete.is-info::before, +button.delete.is-info::after { +    background-color: #237fbd; +} + +/* Give outlines to active tags */ +span.filter-box-tag, +span.resource-tag.active { +    outline-width: 1px; +    outline-style: solid; +} + +/* Disable transitions */ +.no-transition { +  -webkit-transition: none !important; +  -moz-transition: none !important; +  -o-transition: none !important; +  transition: none !important; +} + +/* Make filter tags sparkle when selected! */ +@keyframes glow_success { +    from { box-shadow: 0 0 2px 2px #aef4af; } +    33% { box-shadow: 0 0 2px 2px #87af7a; } +    66% { box-shadow: 0 0 2px 2px #9ceaac; } +    to { box-shadow: 0 0 2px 2px #7cbf64; } +} + +@keyframes glow_primary { +    from { box-shadow: 0 0 2px 2px #aeb8f3; } +    33% { box-shadow: 0 0 2px 2px #909ed9; } +    66% { box-shadow: 0 0 2px 2px #6d7ed4; } +    to { box-shadow: 0 0 2px 2px #6383b3; } +} + +@keyframes glow_danger { +    from { box-shadow: 0 0 2px 2px #c9495f; } +    33% { box-shadow: 0 0 2px 2px #92486f; } +    66% { box-shadow: 0 0 2px 2px #d455ba; } +    to { box-shadow: 0 0 2px 2px #ff8192; } +} +@keyframes glow_info { +    from { box-shadow: 0 0 2px 2px #4592c9; } +    33% { box-shadow: 0 0 2px 2px #6196bb; } +    66% { box-shadow: 0 0 2px 2px #5adade; } +    to { box-shadow: 0 0 2px 2px #6bcfdc; } +} + +span.resource-tag.active.is-primary { +    animation: glow_primary 4s infinite alternate; +} +span.resource-tag.active.has-background-danger-light { +    animation: glow_danger 4s infinite alternate; +} +span.resource-tag.active.has-background-success-light { +    animation: glow_success 4s infinite alternate; +} +span.resource-tag.active.has-background-info-light { +    animation: glow_info 4s infinite alternate;  } -#communitiesBlock { -    background-image: linear-gradient(141deg, #3b756f 0%, #3a847c 71%, #41948b 100%); +/* Smaller filter category headers when on mobile */ +@media screen and (max-width: 480px) { +    .filter-category-header .card-header .card-header-title { +        font-size: 14px; +        padding: 0; +    } +    .filter-panel { +        padding-top: 4px; +        padding-bottom: 4px; +    }  } -#podcastsBlock { -    background-image: linear-gradient(141deg, #232382 0%, #30309c 71%, #4343ad 100%); +/* Constrain the width of the filterbox */ +@media screen and (min-width: 769px) { +    .filtering-column { +        max-width: 25rem; +        min-width: 18rem; +    }  } diff --git a/pydis_site/static/css/resources/resources_list.css b/pydis_site/static/css/resources/resources_list.css deleted file mode 100644 index 33129c87..00000000 --- a/pydis_site/static/css/resources/resources_list.css +++ /dev/null @@ -1,55 +0,0 @@ -.breadcrumb-section { -    padding: 1rem; -} - -i.resource-icon.is-orangered { -    color: #FE640A; -} - -i.resource-icon.is-orangered:hover { -    color: #fe9840; -} - -i.resource-icon.is-blurple { -    color: #7289DA; -} - -i.resource-icon.is-blurple:hover { -    color: #93a8da; -} - -i.resource-icon.is-teal { -    color: #95DBE5; -} - -i.resource-icon.is-teal:hover { -    color: #a9f5ff; -} - -i.resource-icon.is-youtube-red { -    color: #BB0000; -} - -i.resource-icon.is-youtube-red:hover { -    color: #f80000; -} - -i.resource-icon.is-amazon-orange { -    color: #FF9900; -} - -i.resource-icon.is-amazon-orange:hover { -    color: #ffb71a; -} - -i.resource-icon.is-black { -    color: #000000; -} - -i.resource-icon.is-black { -    color: #191919; -} - -i.has-icon-padding { -    padding: 0 10px 25px 0; -} diff --git a/pydis_site/static/images/resources/duck_pond_404.jpg b/pydis_site/static/images/resources/duck_pond_404.jpgBinary files differ new file mode 100644 index 00000000..29bcf1d6 --- /dev/null +++ b/pydis_site/static/images/resources/duck_pond_404.jpg diff --git a/pydis_site/static/js/collapsibles.js b/pydis_site/static/js/collapsibles.js new file mode 100644 index 00000000..1df0b9fe --- /dev/null +++ b/pydis_site/static/js/collapsibles.js @@ -0,0 +1,67 @@ +/* +A utility for creating simple collapsible cards. + +To see this in action, go to /resources or /pages/guides/pydis-guides/contributing/bot/ + +// HOW TO USE THIS // +First, import this file and the corresponding css file into your template. + +  <link rel="stylesheet" href="{% static "css/collapsibles.css" %}"> +  <script defer src="{% static "js/collapsibles.js" %}"></script> + +Next, you'll need some HTML that these scripts can interact with. + +<div class="card"> +    <button type="button" class="card-header collapsible"> +        <span class="card-header-title subtitle is-6 my-2 ml-2">Your headline</span> +        <span class="card-header-icon"> +            <i class="fas fa-fw fa-angle-down title is-5" aria-hidden="true"></i> +        </span> +    </button> +    <div class="collapsible-content collapsed"> +        <div class="card-content"> +            You can put anything you want here. Lists, more divs, flexboxes, images, whatever. +        </div> +    </div> +</div> + +That's it! Collapsing stuff should now work. + */ + +document.addEventListener("DOMContentLoaded", () => { +    const contentContainers = document.getElementsByClassName("collapsible-content"); +    for (const container of contentContainers) { +        // Close any collapsibles that are marked as initially collapsed +        if (container.classList.contains("collapsed")) { +            container.style.maxHeight = "0px"; +        // Set maxHeight to the size of the container on all other containers. +        } else { +            container.style.maxHeight = container.scrollHeight + "px"; +        } +    } + +    // Listen for click events, and collapse or explode +    const headers = document.getElementsByClassName("collapsible"); +    for (const header of headers) { +        const content = header.nextElementSibling; +        const icon = header.querySelector(".card-header-icon i"); + +        // Any collapsibles that are not initially collapsed needs an icon switch. +        if (!content.classList.contains("collapsed")) { +            icon.classList.remove("fas", "fa-angle-down"); +            icon.classList.add("far", "fa-window-minimize"); +        } + +        header.addEventListener("click", () => { +            if (content.style.maxHeight !== "0px"){ +                content.style.maxHeight = "0px"; +                icon.classList.remove("far", "fa-window-minimize"); +                icon.classList.add("fas", "fa-angle-down"); +            } else { +                content.style.maxHeight = content.scrollHeight + "px"; +                icon.classList.remove("fas", "fa-angle-down"); +                icon.classList.add("far", "fa-window-minimize"); +            } +        }); +    } +}); diff --git a/pydis_site/static/js/content/page.js b/pydis_site/static/js/content/page.js deleted file mode 100644 index 366a033c..00000000 --- a/pydis_site/static/js/content/page.js +++ /dev/null @@ -1,13 +0,0 @@ -document.addEventListener("DOMContentLoaded", () => { -    const headers = document.getElementsByClassName("collapsible"); -    for (const header of headers) { -        header.addEventListener("click", () => { -            var content = header.nextElementSibling; -            if (content.style.maxHeight){ -              content.style.maxHeight = null; -            } else { -              content.style.maxHeight = content.scrollHeight + "px"; -            } -        }); -    } -}); diff --git a/pydis_site/static/js/resources/resources.js b/pydis_site/static/js/resources/resources.js new file mode 100644 index 00000000..508849e1 --- /dev/null +++ b/pydis_site/static/js/resources/resources.js @@ -0,0 +1,285 @@ +"use strict"; + +// Filters that are currently selected +var activeFilters = { +    topics: [], +    type: [], +    "payment-tiers": [], +    difficulty: [] +}; + +/* Add a filter, and update the UI */ +function addFilter(filterName, filterItem) { +    var filterIndex = activeFilters[filterName].indexOf(filterItem); +    if (filterIndex === -1) { +        activeFilters[filterName].push(filterItem); +    } +    updateUI(); +} + +/* Remove all filters, and update the UI */ +function removeAllFilters() { +    activeFilters = { +        topics: [], +        type: [], +        "payment-tiers": [], +        difficulty: [] +    }; +    updateUI(); +} + +/* Remove a filter, and update the UI */ +function removeFilter(filterName, filterItem) { +    var filterIndex = activeFilters[filterName].indexOf(filterItem); +    if (filterIndex !== -1) { +        activeFilters[filterName].splice(filterIndex, 1); +    } +    updateUI(); +} + +/* Check if there are no filters */ +function noFilters() { +    return ( +        activeFilters.topics.length === 0 && +        activeFilters.type.length === 0 && +        activeFilters["payment-tiers"].length === 0 && +        activeFilters.difficulty.length === 0 +    ); +} + +/* Get the params out of the URL and use them. This is run when the page loads. */ +function deserializeURLParams() { +    let searchParams = new window.URLSearchParams(window.location.search); + +    // Work through the parameters and add them to the filter object +    $.each(Object.keys(activeFilters), function(_, filterType) { +        let paramFilterContent = searchParams.get(filterType); + +        if (paramFilterContent !== null) { +            // We use split here because we always want an array, not a string. +            let paramFilterArray = paramFilterContent.split(","); + +            // Update the corresponding filter UI, so it reflects the internal state. +            let filterAdded = false; +            $(paramFilterArray).each(function(_, filter) { +                // Make sure the filter is valid before we do anything. +                if (String(filter) === "rickroll" && filterType === "type") { +                    window.location.href = "https://www.youtube.com/watch?v=dQw4w9WgXcQ"; +                } else if (String(filter) === "sneakers" && filterType === "topics") { +                    window.location.href = "https://www.youtube.com/watch?v=NNZscmNE9QI"; +                } else if (validFilters.hasOwnProperty(filterType) && validFilters[filterType].includes(String(filter))) { +                    let checkbox = $(`.filter-checkbox[data-filter-name='${filterType}'][data-filter-item='${filter}']`); +                    let filterTag = $(`.filter-box-tag[data-filter-name='${filterType}'][data-filter-item='${filter}']`); +                    let resourceTags = $(`.resource-tag[data-filter-name='${filterType}'][data-filter-item='${filter}']`); +                    checkbox.prop("checked", true); +                    filterTag.show(); +                    resourceTags.addClass("active"); +                    activeFilters[filterType].push(filter); +                    filterAdded = true; +                } +            }); + +            // Ditch all the params from the URL, and recalculate the URL params +            updateURL(); + +            // If we've added a filter, hide stuff +            if (filterAdded) { +                $(".no-tags-selected.tag").hide(); +                $(".close-filters-button").show(); +            } +        } +    }); +} + +/* Update the URL with new parameters */ +function updateURL() { +    // If there's nothing in the filters, we don't want anything in the URL. +    if (noFilters()) { +        window.history.replaceState(null, document.title, './'); +        return; +    } + +    // Iterate through and get rid of empty ones +    let searchParams = new URLSearchParams(activeFilters); +    $.each(activeFilters, function(filterType, filters) { +        if (filters.length === 0) { +            searchParams.delete(filterType); +        } +    }); + +    // Now update the URL +    window.history.replaceState(null, document.title, `?${searchParams.toString()}`); +} + +/* Update the resources to match 'active_filters' */ +function updateUI() { +    let resources = $('.resource-box'); +    let filterTags = $('.filter-box-tag'); +    let resourceTags = $('.resource-tag'); +    let noTagsSelected = $(".no-tags-selected.tag"); +    let closeFiltersButton = $(".close-filters-button"); + +    // Update the URL to match the new filters. +    updateURL(); + +    // If there's nothing in the filters, we can return early. +    if (noFilters()) { +        resources.show(); +        filterTags.hide(); +        noTagsSelected.show(); +        closeFiltersButton.hide(); +        resourceTags.removeClass("active"); +        $(`.filter-checkbox:checked`).prop("checked", false); +        $(".no-resources-found").hide(); +        return; +    } else { +        // Hide everything +        $('.filter-box-tag').hide(); +        $('.resource-tag').removeClass("active"); +        noTagsSelected.show(); +        closeFiltersButton.hide(); + +        // Now conditionally show the stuff we want +        $.each(activeFilters, function(filterType, filters) { +            $.each(filters, function(index, filter) { +                // Show a corresponding filter box tag +                $(`.filter-box-tag[data-filter-name=${filterType}][data-filter-item=${filter}]`).show(); + +                // Make corresponding resource tags active +                $(`.resource-tag[data-filter-name=${filterType}][data-filter-item=${filter}]`).addClass("active"); + +                // Hide the "No filters selected" tag. +                noTagsSelected.hide(); + +                // Show the close filters button +                closeFiltersButton.show(); +            }); +        }); +    } + +    // Otherwise, hide everything and then filter the resources to decide what to show. +    let hasMatches = false; +    resources.hide(); +    resources.filter(function() { +        let validation = { +            topics: false, +            type: false, +            'payment-tiers': false, +            difficulty: false +        }; +        let resourceBox = $(this); + +        // Validate the filters +        $.each(activeFilters, function(filterType, filters) { +            // If the filter list is empty, this passes validation. +            if (filters.length === 0) { +                validation[filterType] = true; +                return; +            } + +            // Otherwise, we need to check if one of the classes exist. +            $.each(filters, function(index, filter) { +                if (resourceBox.hasClass(`${filterType}-${filter}`)) { +                    validation[filterType] = true; +                } +            }); +        }); + +        // If validation passes, show the resource. +        if (Object.values(validation).every(Boolean)) { +            hasMatches = true; +            return true; +        } else { +            return false; +        } +    }).show(); + + +    // If there are no matches, show the no matches message +    if (!hasMatches) { +        $(".no-resources-found").show(); +    } else { +        $(".no-resources-found").hide(); +    } +} + +// Executed when the page has finished loading. +document.addEventListener("DOMContentLoaded", function () { +    /* Check if the user has navigated to one of the old resource pages, +       like pydis.com/resources/communities. In this case, we'll rewrite +       the URL before we do anything else. */ +    let resourceTypeInput = $("#resource-type-input").val(); +    if (resourceTypeInput !== "None") { +        window.history.replaceState(null, document.title, `../?type=${resourceTypeInput}`); +    } + +    // Update the filters on page load to reflect URL parameters. +    $('.filter-box-tag').hide(); +    deserializeURLParams(); +    updateUI(); + +    // If this is a mobile device, collapse all the categories to win back some screen real estate. +    if (screen.width < 480) { +        let categoryHeaders = $(".filter-category-header .collapsible-content"); +        let icons = $('.filter-category-header button .card-header-icon i'); +        categoryHeaders.addClass("no-transition collapsed"); +        icons.removeClass(["far", "fa-window-minimize"]); +        icons.addClass(["fas", "fa-angle-down"]); + +        // Wait 10ms before removing this class, or else the transition will animate due to a race condition. +        setTimeout(() => { categoryHeaders.removeClass("no-transition"); }, 10); +    } + +    // If you click on the div surrounding the filter checkbox, it clicks the corresponding checkbox. +    $('.filter-panel').on("click",function(event) { +        let hitsCheckbox = Boolean(String(event.target)); + +        if (!hitsCheckbox) { +            let checkbox = $(this).find(".filter-checkbox"); +            checkbox.prop("checked", !checkbox.prop("checked")); +            checkbox.trigger("change"); +        } +    }); + +    // If you click on one of the tags in the filter box, it unchecks the corresponding checkbox. +    $('.filter-box-tag').on("click", function() { +        let filterItem = this.dataset.filterItem; +        let filterName = this.dataset.filterName; +        let checkbox = $(`.filter-checkbox[data-filter-name='${filterName}'][data-filter-item='${filterItem}']`); + +        removeFilter(filterName, filterItem); +        checkbox.prop("checked", false); +    }); + +    // If you click on one of the tags in the resource cards, it clicks the corresponding checkbox. +    $('.resource-tag').on("click", function() { +        let filterItem = this.dataset.filterItem; +        let filterName = this.dataset.filterName; +        let checkbox = $(`.filter-checkbox[data-filter-name='${filterName}'][data-filter-item='${filterItem}']`); + +        if (!$(this).hasClass("active")) { +            addFilter(filterName, filterItem); +            checkbox.prop("checked", true); +        } else { +            removeFilter(filterName, filterItem); +            checkbox.prop("checked", false); +        } +    }); + +    // When you click the little gray x, remove all filters. +    $(".close-filters-button").on("click", function() { +        removeAllFilters(); +    }); + +    // When checkboxes are toggled, trigger a filter update. +    $('.filter-checkbox').on("change", function (event) { +        let filterItem = this.dataset.filterItem; +        let filterName = this.dataset.filterName; + +        if (this.checked && !activeFilters[filterName].includes(filterItem)) { +            addFilter(filterName, filterItem); +        } else if (!this.checked && activeFilters[filterName].includes(filterItem)) { +            removeFilter(filterName, filterItem); +        } +    }); +}); diff --git a/pydis_site/templates/base/base.html b/pydis_site/templates/base/base.html index 906fc577..b7322f12 100644 --- a/pydis_site/templates/base/base.html +++ b/pydis_site/templates/base/base.html @@ -24,10 +24,7 @@    <title>Python Discord | {% block title %}Website{% endblock %}</title>    {% bulma %} - -  {# Font-awesome here is defined explicitly so that we can have Pro #} -  <script src="https://kit.fontawesome.com/ae6a3152d8.js"></script> - +  {% font_awesome %}    <link rel="stylesheet" href="{% static "css/base/base.css" %}">    {% block head %}{% endblock %} diff --git a/pydis_site/templates/base/navbar.html b/pydis_site/templates/base/navbar.html index 4b68dd6c..d7fb4f4c 100644 --- a/pydis_site/templates/base/navbar.html +++ b/pydis_site/templates/base/navbar.html @@ -67,9 +67,6 @@            <a class="navbar-item" href="{% url "resources:index" %}">              Resources            </a> -          <a class="navbar-item" href="{% url "resources:resources" category="tools" %}"> -            Tools -          </a>            <a class="navbar-item" href="{% url "events:index" %}">              Events            </a> @@ -79,6 +76,9 @@            <a class="navbar-item" href="{% url "content:page_category" location="frequently-asked-questions" %}">              FAQ            </a> +          <a class="navbar-item" href="{% url "content:page_category" location="guides" %}"> +            Guides +          </a>            <a class="navbar-item" href="{% url 'home:timeline' %}">              Timeline            </a> diff --git a/pydis_site/templates/content/base.html b/pydis_site/templates/content/base.html index 00f4fce4..4a19a275 100644 --- a/pydis_site/templates/content/base.html +++ b/pydis_site/templates/content/base.html @@ -7,7 +7,8 @@      <meta property="og:type" content="website" />      <meta property="og:description" content="{{ page_description }}" />      <link rel="stylesheet" href="{% static "css/content/page.css" %}"> -    <script src="{% static "js/content/page.js" %}"></script> +    <link rel="stylesheet" href="{% static "css/collapsibles.css" %}"> +    <script src="{% static "js/collapsibles.js" %}"></script>  {% endblock %}  {% block content %} diff --git a/pydis_site/templates/resources/resource_box.html b/pydis_site/templates/resources/resource_box.html index af7c8d65..e26203e9 100644 --- a/pydis_site/templates/resources/resource_box.html +++ b/pydis_site/templates/resources/resource_box.html @@ -1,6 +1,8 @@  {% load as_icon %} +{% load to_kebabcase %} +{% load get_category_icon %} -<div class="box" style="max-width: 800px;"> +<div class="box resource-box {{ resource.css_classes }}">      {% if 'title_url' in resource %}          <a href="{{ resource.title_url }}">              {% include "resources/resource_box_header.html" %} @@ -9,14 +11,69 @@          {% include "resources/resource_box_header.html" %}      {% endif %} -    <p class="is-italic">{{ resource.description|safe }}</p> +    <p>{{ resource.description|safe }}</p> -    {# Icons #} -    {% for icon in resource.urls %} -        <span class="icon is-size-4 is-medium" style="margin: 5px;"> -            <a href="{{ icon.url }}"> -                <i class="{{ icon.icon|as_icon }} is-size-3 resource-icon is-{{ icon.color }}"></i> -            </a> -        </span> -    {% endfor %} +    <div class="is-flex is-align-items-center"> +        {# Add primary link #} +        {% if "title_url" in resource %} +            <span class="icon is-size-4" style="margin: 5px;"> +                <a href="{{ resource.title_url }}"> +                    <i class="fas fa-external-link-alt fa-fw is-size-4 resource-icon is-hoverable is-primary"></i> +                </a> +            </span> +        {% endif %} + +        {# Add all additional icon #} +        {% for icon in resource.urls %} +            <span class="icon is-size-4" style="margin: 5px;"> +                <a href="{{ icon.url }}"> +                    <i class="{{ icon.icon|as_icon }} fa-fw is-size-4 resource-icon is-hoverable is-{{ icon.color }}"></i> +                </a> +            </span> +        {% endfor %} + +        {#  Tags #} +        <div class="resource-tag-container is-flex ml-auto is-flex-wrap-wrap is-justify-content-end"> +            {% for tag in resource.tags.topics %} +            <span +                class="tag resource-tag is-primary is-light ml-2 mt-2" +                data-filter-name="topics" +                data-filter-item="{{ tag|to_kebabcase }}" +            > +                <i class="{{ tag|title|get_category_icon }} mr-1"></i> +                {{ tag|title }} +            </span> +            {% endfor %} +            {% for tag in resource.tags.type %} +            <span +                class="tag resource-tag has-background-success-light has-text-success-dark ml-2 mt-2" +                data-filter-name="type" +                data-filter-item="{{ tag|to_kebabcase }}" +            > +                <i class="{{ tag|title|get_category_icon }} mr-1"></i> +                {{ tag|title }} +            </span> +            {% endfor %} +            {% for tag in resource.tags.payment_tiers %} +            <span +                class="tag resource-tag has-background-danger-light has-text-danger-dark ml-2 mt-2" +                data-filter-name="payment-tiers" +                data-filter-item="{{ tag|to_kebabcase }}" +            > +                <i class="{{ tag|title|get_category_icon }} mr-1"></i> +                {{ tag|title }} +            </span> +            {% endfor %} +            {% for tag in resource.tags.difficulty %} +            <span +                class="tag resource-tag has-background-info-light has-text-info-dark ml-2 mt-2" +                data-filter-name="difficulty" +                data-filter-item="{{ tag|to_kebabcase }}" +            > +                <i class="{{ tag|title|get_category_icon }} mr-1"></i> +                {{ tag|title }} +            </span> +            {% endfor %} +        </div> +    </div>  </div> diff --git a/pydis_site/templates/resources/resources.html b/pydis_site/templates/resources/resources.html index f1f487cf..70fad097 100644 --- a/pydis_site/templates/resources/resources.html +++ b/pydis_site/templates/resources/resources.html @@ -1,90 +1,164 @@  {% extends 'base/base.html' %} +{% load as_icon %} +{% load to_kebabcase %} +{% load get_category_icon %}  {% load static %}  {% block title %}Resources{% endblock %}  {% block head %} -  <link rel="stylesheet" href="{% static "css/resources/resources.css" %}"> +    {# Inject a JSON object of all valid filter types from the view #} +    <script> +        const validFilters = {{ valid_filters | safe }} +    </script> + +    <link rel="stylesheet" href="{% static "css/resources/resources.css" %}"> +    <link rel="stylesheet" href="{% static "css/collapsibles.css" %}"> +    <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.6.0/jquery.min.js"></script> +    <script defer src="{% static "js/resources/resources.js" %}"></script> +    <script defer src="{% static "js/collapsibles.js" %}"></script>  {% endblock %}  {% block content %} -  {% include "base/navbar.html" %} +    {% include "base/navbar.html" %} +    <input type="hidden" id="resource-type-input" value="{{ resource_type }}"> +    <section class="section"> +        <div class="columns is-variable is-6 is-centered"> +            {# Filtering toolbox #} +            <div class="column filtering-column is-one-third"> +                <div class="content is-justify-content-center"> +                    <nav id="resource-filtering-panel" class="panel is-primary"> +                        <p class="panel-heading has-text-centered" id="filter-panel-header">Filter Resources</p> + +                        {# Filter box tags #} +                        <div class="card filter-tags"> +                            <div class="is-flex ml-auto"> +                                <div> +                                    {# A filter tag for when there are no filters active #} +                                    <span class="no-tags-selected tag has-background-disabled has-text-disabled ml-2 mt-2"> +                                            <i class="fas fa-ban mr-1"></i> +                                            No filters selected +                                    </span> + +                                    {% for filter_name, filter_data in filters.items %} +                                        {% for filter_item in filter_data.filters %} +                                            {% if filter_name == "Difficulty" %} +                                                <span +                                                        class="filter-box-tag tag has-background-info-light has-text-info-dark ml-2 mt-2" +                                                        data-filter-name="{{ filter_name|to_kebabcase }}" +                                                        data-filter-item="{{ filter_item|to_kebabcase }}" +                                                > +                                                        <i class="{{ filter_item|title|get_category_icon }} mr-1"></i> +                                                        {{ filter_item|title }} +                                                        <button class="delete is-small is-info has-background-info-light"></button> +                                                    </span> +                                            {% endif %} +                                            {% if filter_name == "Type" %} +                                                <span +                                                        class="filter-box-tag tag has-background-success-light has-text-success-dark ml-2 mt-2" +                                                        data-filter-name="{{ filter_name|to_kebabcase }}" +                                                        data-filter-item="{{ filter_item|to_kebabcase }}" +                                                > +                                                        <i class="{{ filter_item|title|get_category_icon }} mr-1"></i> +                                                        {{ filter_item|title }} +                                                        <button class="delete is-small is-success has-background-success-light"></button> +                                                    </span> +                                            {% endif %} +                                            {% if filter_name == "Payment tiers" %} +                                                <span +                                                        class="filter-box-tag tag has-background-danger-light has-text-danger-dark ml-2 mt-2" +                                                        data-filter-name="{{ filter_name|to_kebabcase }}" +                                                        data-filter-item="{{ filter_item|to_kebabcase }}" +                                                > +                                                        <i class="{{ filter_item|title|get_category_icon }} mr-1"></i> +                                                        {{ filter_item|title }} +                                                        <button class="delete is-small is-danger has-background-danger-light"></button> +                                                    </span> +                                            {% endif %} +                                            {% if filter_name == "Topics" %} +                                                <span +                                                        class="filter-box-tag tag is-primary is-light ml-2 mt-2" +                                                        data-filter-name="{{ filter_name|to_kebabcase }}" +                                                        data-filter-item="{{ filter_item|to_kebabcase }}" +                                                > +                                                        <i class="{{ filter_item|title|get_category_icon }} mr-1"></i> +                                                        {{ filter_item|title }} +                                                        <button class="delete is-small is-primary has-background-primary-light"></button> +                                                    </span> +                                            {% endif %} +                                        {% endfor %} +                                    {% endfor %} +                                </div> +                                <div class="close-filters-button"> +                                    {# A little x in the top right, visible only when filters are active, which removes all filters. #} +                                    <a class="icon"> +                                        <i class="fas fa-window-close"></i> +                                    </a> -  <section class="section"> -      <div class="container"> -          <div class="content"> -              <h1>Resources</h1> +                                </div> +                            </div> +                        </div> -              <div class="tile is-ancestor"> -                  <a class="tile is-parent" href="{% url "content:page_category" location="guides" %}"> -                      <article class="tile is-child box hero is-primary is-bold"> -                          <p class="title is-size-1"><i class="fad fa-info-circle" aria-hidden="true"></i> Guides</p> -                          <p class="subtitle is-size-4">Made by us, for you</p> -                      </article> -                  </a> +                        {# Filter checkboxes #} +                        {% for filter_name, filter_data in filters.items %} +                            <div class="card filter-category-header"> +                                <button type="button" class="card-header collapsible"> +                                    <span class="card-header-title subtitle is-6 my-2 ml-2"> +                                        <i class="fa-fw {{ filter_data.icon }} is-primary" aria-hidden="true"></i>  {{ filter_name }} +                                    </span> +                                    <span class="card-header-icon"> +                                        {% if not filter_data.hidden %} +                                            <i class="far fa-fw fa-window-minimize is-6 title" aria-hidden="true"></i> +                                        {% else %} +                                            <i class="fas fa-fw fa-angle-down is-6 title" aria-hidden="true"></i> +                                        {% endif %} +                                    </span> +                                </button> -                  <div class="tile is-vertical is-9"> -                      <div class="tile"> -                          <a class="tile is-8 is-parent" href="{% url "resources:resources" category="reading" %}"> -                              <article class="tile is-child box hero is-black" id="readingBlock"> -                                  <p class="title is-size-1"><i class="fad fa-book-alt" aria-hidden="true"></i> Read</p> -                                  <p class="subtitle is-size-4">Lovingly curated books to explore</p> -                              </article> -                          </a> +                            {# Checkboxes #} +                            {% if filter_data.hidden %} +                                <div class="collapsible-content collapsed"> +                            {% else %} +                                <div class="collapsible-content"> +                            {% endif %} +                                    <div class="card-content"> +                                        {% for filter_item in filter_data.filters %} +                                            <a class="panel-block filter-panel"> +                                                <label class="checkbox"> +                                                    <input +                                                            class="filter-checkbox" +                                                            type="checkbox" +                                                            data-filter-name="{{ filter_name|to_kebabcase }}" +                                                            data-filter-item="{{ filter_item|to_kebabcase }}" +                                                    > +                                                    {{ filter_item }} +                                                </label> +                                            </a> +                                        {% endfor %} +                                    </div> +                                </div> +                            </div> +                        {% endfor %} +                    </nav> +                </div> +            </div> -                          <div class="tile"> -                              <a class="tile is-parent" href="{% url "resources:resources" category="videos" %}"> -                                  <article class="tile is-child box hero is-danger is-bold"> -                                      <p class="title is-size-1"><i class="fad fa-video" aria-hidden="true"></i> Watch</p> -                                      <p class="subtitle is-size-4">Visually engaging</p> -                                  </article> -                              </a> -                          </div> -                      </div> +            <div class="column is-two-thirds"> +                {# Message to display when there are no hits #} +                <div class="no-resources-found"> +                    <h2 class="title is-3 has-text-centered pt-0 pb-6">No matching resources found!</h2> +                    <img src="{% static "images/resources/duck_pond_404.jpg" %}"> +                </div> -                      <div class="tile"> -                          <a class="tile is-parent" href="{% url "resources:resources" category="interactive" %}"> -                              <article class="tile is-child box hero is-black" id="interactiveBlock"> -                                  <p class="title is-size-1"><i class="fad fa-code" aria-hidden="true"></i> Try</p> -                                  <p class="subtitle is-size-4">Interactively discover the possibilities</p> -                              </article> -                          </a> -                          <a class="tile is-8 is-parent" href="{% url "resources:resources" category="courses" %}"> -                              <article class="tile is-child box hero is-success is-bold"> -                                  <p class="title is-size-1"><i class="fad fa-graduation-cap" aria-hidden="true"></i> Learn</p> -                                  <p class="subtitle is-size-4">Structured courses with clear goals</p> -                              </article> -                          </a> -                      </div> -                  </div> -              </div> -              <div class="tile is-ancestor"> -                  <div class="tile is-vertical is-9"> -                      <div class="tile"> -                          <a class="tile is-8 is-parent" href="{% url "resources:resources" category="communities" %}"> -                              <article class="tile is-child box hero is-black" id="communitiesBlock"> -                                  <p class="title is-size-1"><i class="fad fa-users" aria-hidden="true"></i> Communities</p> -                                  <p class="subtitle is-size-4">Some of our best friends</p> -                              </article> -                          </a> -                          <div class="tile"> -                              <a class="tile is-parent" href="{% url "resources:resources" category="podcasts" %}"> -                                  <article class="tile is-child box hero is-black" id="podcastsBlock"> -                                      <p class="title is-size-1"><i class="fad fa-podcast" aria-hidden="true"></i> Listen</p> -                                      <p class="subtitle is-size-4">Regular podcasts to follow</p> -                                  </article> -                              </a> -                          </div> -                      </div> -                  </div> -                  <a class="tile is-parent" href="{% url "resources:resources" category="tools" %}"> -                      <article class="tile is-child box hero is-dark"> -                          <p class="title is-size-1"><i class="fad fa-tools" aria-hidden="true"></i> Tools</p> -                          <p class="subtitle is-size-4">Things we love to use</p> -                      </article> -                  </a> -              </div> -          </div> -      </div> -  </section> +                {# Resource cards #} +                <div class="content is-flex is-justify-content-center"> +                    <div> +                        {% for resource in resources.values %} +                            {% include "resources/resource_box.html" %} +                        {% endfor %} +                    </div> +                </div> +            </div> +        </div> +    </section>  {% endblock %} diff --git a/pydis_site/templates/resources/resources_list.html b/pydis_site/templates/resources/resources_list.html deleted file mode 100644 index e2be3cb7..00000000 --- a/pydis_site/templates/resources/resources_list.html +++ /dev/null @@ -1,52 +0,0 @@ -{% extends "base/base.html" %} -{% load as_icon %} -{% load static %} - -{% block title %}{{ category_info.name }}{% endblock %} -{% block head %} -    <link rel="stylesheet" href="{% static "css/resources/resources_list.css" %}"> -{% endblock %} - -{% block content %} -    {% include "base/navbar.html" %} - -    <section class="section breadcrumb-section"> -        <div class="container"> -            <nav class="breadcrumb is-pulled-left" aria-label="breadcrumbs"> -                <ul> -                    <li><a href="{% url "resources:index" %}">Resources</a></li> -                    <li class="is-active"><a href="#">{{ category_info.name }}</a></li> -                </ul> -            </nav> -        </div> -    </section> - -    <section class="section"> -        <div class="container"> -            <div class="content"> -                <h1>{{ category_info.name }}</h1> -                <p>{{ category_info.description|safe }}</p> -                <div> -                    {% for resource in resources|dictsort:"position" %} -                        {% include "resources/resource_box.html" %} -                    {% endfor %} - -                    {% for subcategory in subcategories|dictsort:"category_info.position" %} -                        <h2 id="{{ subcategory.category_info.raw_name }}"> -                            <a href="{% url "resources:resources" category=category_info.raw_name %}#{{ subcategory.category_info.raw_name }}"> -                                {{ subcategory.category_info.name }} -                            </a> -                        </h2> -                        <p>{{ subcategory.category_info.description|safe }}</p> - -                        {% for resource in subcategory.resources|dictsort:"position" %} -                            {% with category_info=subcategory.category_info %} -                                {% include "resources/resource_box.html" %} -                            {% endwith %} -                        {% endfor %} -                    {% endfor %} -                </div> -            </div> -        </div> -    </section> -{% endblock %} diff --git a/pyproject.toml b/pyproject.toml index 3fa22740..b350836e 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -12,7 +12,7 @@ django-environ = "~=0.4.5"  django-filter = "~=21.1"  djangorestframework = "~=3.12.0"  psycopg2-binary = "~=2.8.0" -django-simple-bulma = "~=2.1" +django-simple-bulma = "~=2.4"  whitenoise = "~=5.0"  requests = "~=2.21"  pyyaml = "~=5.1" @@ -51,6 +51,7 @@ start = "python manage.py run --debug"  makemigrations = "python manage.py makemigrations"  django_shell = "python manage.py shell"  test = "coverage run manage.py test" +coverage = "coverage run manage.py test --no-input; coverage report -m"  report = "coverage report -m"  lint = "pre-commit run --all-files"  precommit = "pre-commit install" | 
