Criando e renderizando estrutura com anos e meses em Django
-
27-09-2019 - |
Pergunta
No meu aplicativo de blog, preciso de uma estrutura (criada como uma variável no processador de contexto) que armazenará o número de meses e o ano correspondente de 5 meses consecutivos até o atual. Portanto, se o mês atual for de dezembro, teremos ano: 2010 e meses: 12,11,10,9,8. Se o mês será em janeiro, teremos anos 2010: meses: 1 e anos: 2009 meses: 12, 11, 10, 9. Meu objetivo é mostrar um arquivo na seguinte forma:
- 2010
- January
- 2009
- December
- November
- October
- September
Como criá -lo e que estrutura devo usar? E então como mostrá -lo? Eu acho que preciso de alguma estrutura aninhada, mas que será possível renderizar no django <1,2?
Eu comecei sozinho, mas me perdi completamente em algum momento:
now = datetime.datetime.now()
years = []
months = []
archive = []
if now.month in range(5, 12, 1):
months = range(now.month, now.month-5, -1)
if months:
years = now.year
else:
diff = 5 - now.month
for i in range(1, now.month, 1):
archive.append({
"month": i,
"year": now.year,
})
for i in range(0, diff, 1):
tmpMonth = 12 - int(i)
archive.append({
"month": tmpMonth,
"year": now.year-1,
})
if archive:
years = [now.year, now.year-1]
Solução
Como criá -lo e que estrutura devo usar?
Eu iria com uma lista de tuplas do mês. Aqui está uma implementação de amostra. Você precisará do prático Python-Dateutil biblioteca para fazer isso funcionar.
from datetime import datetime
from dateutil.relativedelta import relativedelta
def get_5_previous_year_months(a_day):
"""Returns a list of year, month tuples for the current and previous
5 months relative to a_day"""
current_year, current_month = a_day.year, a_day.month
first_of_month = datetime(current_year, current_month, 1)
previous_months = (first_of_month - relativedelta(months = months)
for months in range(0, 5))
return ((pm.year, pm.month) for pm in previous_months)
def get_current_and_5_previous_months():
return get_5_previous_year_months(datetime.today())
E então como mostrá -lo?
Aqui está uma maneira muito simplista de mostrá -lo. Eu acho que você pode limpá -lo substituindo o <ul>
elementos com <div>
e estilizando -o adequadamente.
<ul>
{% for year, month in previous_year_months %}
{% ifchanged year %}
</ul><li>{{ year }}</li><ul>
{% endifchanged %}
<li>{{ month }}</li>
{% endfor %}
</ul>
Onde previous_year_months
é uma variável de contexto correspondente ao resultado retornado por get_current_and_5_previous_months
.