-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathcalgen.py
269 lines (223 loc) · 8.12 KB
/
calgen.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
# Don't Break the Chain calendar
#
# Design based on:
# http://karenkavett.com/blog/3555/dont-break-the-chain-calendar-2017-free-motivational-tool-printable.php
from datetime import date, timedelta
from copy import copy
from reportlab.pdfgen import canvas
from reportlab.lib.pagesizes import letter, A4
COLOR_BLACK = "#000000"
COLOR_GRAY = "#777777"
def add_one_year(d):
"""Calculates the number of days until the same date next year."""
same_day_next_year = d + timedelta(days=365)
# Starting on the leap day?
if (d.month, d.day) == (2, 29):
return same_day_next_year
# Does the time period include a leap year?
if same_day_next_year.day != d.day:
same_day_next_year += timedelta(days=1)
return same_day_next_year
class CalendarDrawing:
def __init__(self, start_date=None, end_date=None, label_side="left", week_start=0):
"""
:param start_date: Date the calendar starts on
:param end_date: Date the calendar ends on
:param label_side: ???
:param week_start: The day of the week, 0-6 being Monday-Sunday, which
weeks start.
"""
self.start_date = date.today() if start_date is None else start_date
if end_date is None:
# Default to one year, correcting for leap year.
self.end_date = add_one_year(self.start_date) - timedelta(days=1)
else:
self.end_date = end_date
self.label_side = label_side
self.week_start = week_start
self.font_size = 12
self.col_spacing = self.font_size * 2.0
self.row_spacing = self.font_size * 1.7
def get_day_pos(self, d):
"""Return the column this day should be placed in.
:returns: 0-6, with 0 being the first day of the week according to `self.week_start`.
"""
return (d.weekday() - self.week_start) % 7
def get_width(self):
"""Does not include side labels."""
return self.col_spacing * 7
def get_month_color(self, d):
if (d.year*12 + d.month) % 2 == 0:
return COLOR_BLACK
else:
return COLOR_GRAY
def set_font_style(self, canvas, style, d):
if style == "day":
canvas.setFont("Helvetica", self.font_size)
canvas.setFillColor(self.get_month_color(d))
elif style == "month":
canvas.setFont("Helvetica", self.font_size)
canvas.setFillColor(self.get_month_color(d))
elif style == "year":
canvas.setFont("Helvetica", self.font_size*1.1)
canvas.setFillColor(self.get_month_color(d))
else:
raise ValueError()
def weeks(self):
d = self.start_date
row = 0
week = []
while d <= self.end_date:
week.append(d)
d += timedelta(days=1)
if self.get_day_pos(d) == 0:
yield week
week = []
row += 1
if week:
yield week
def draw(self, canvas, x, y, show_year_on_first_row=False):
x += self.col_spacing/2
y += self.row_spacing/2
last_week = None
for row, week in enumerate(self.weeks()):
days_of_month = [d.day for d in week]
# Row label (ex: 2017 Jan)
side = 0 if self.label_side == "left" else -1
if row == 0 or week[side].month != last_week[side].month:
force_show_year = row == 0 and show_year_on_first_row
self.draw_row_label(canvas, x, y, week[side], row, force_show_year)
# Month divider (line between months)
if 1 in days_of_month:
col = self.get_day_pos(week[days_of_month.index(1)])
self.draw_month_divider(canvas, x, y, row, col)
# Days of Month
for d in week:
self.set_font_style(canvas, "day", d)
canvas.drawCentredString(
x + self.col_spacing*self.get_day_pos(d),
y - self.row_spacing*row,
str(d.day)
)
last_week = week
def draw_row_label(self, canvas, x, y, d, row, force_show_year=False):
"""
Draw month on the side. Also draws year if January, or if
`force_show_year` is True.
"""
# Month Text
self.set_font_style(canvas, "month", d)
if self.label_side == "left":
offset = self.col_spacing*(-1.5)
elif self.label_side == "right":
offset = self.col_spacing*(1.5+6)
else:
raise ValueError("Invalid value for label_side")
canvas.drawCentredString(
x + offset,
y - self.row_spacing*row,
d.strftime("%b")
)
# Year Text
if d.month == 1 or force_show_year:
self.set_font_style(canvas, "year", d)
if self.label_side == "left":
offset = self.col_spacing*(-2.9)
elif self.label_side == "right":
offset = self.col_spacing*(2.9+6)
else:
raise ValueError("Invalid value for label_side")
canvas.drawCentredString(
x + offset,
y - self.row_spacing*row,
d.strftime("%Y")
)
def draw_month_divider(self, canvas, x, y, row, col):
canvas.setStrokeColor(COLOR_GRAY)
p = canvas.beginPath()
p.moveTo(
x - self.col_spacing/2,
y-self.row_spacing*row - self.row_spacing/4
)
p.lineTo(
x + col*self.col_spacing - self.col_spacing/2,
y-self.row_spacing*row - self.row_spacing/4
)
if col == 0:
p.moveTo(
x + col*self.col_spacing - self.col_spacing/2,
y-self.row_spacing*(row-1) - self.row_spacing/4
)
else:
p.lineTo(
x + col*self.col_spacing - self.col_spacing/2,
y-self.row_spacing*(row-1) - self.row_spacing/4
)
p.lineTo(
x + 7*self.col_spacing - self.col_spacing/2,
y-self.row_spacing*(row-1) - self.row_spacing/4
)
canvas.drawPath(p)
def split(self):
left = copy(self)
right = copy(self)
weeks = list(self.weeks())
last_week_in_left = weeks[len(weeks)//2 - 1]
first_week_in_right = weeks[len(weeks)//2]
left.end_date = last_week_in_left[-1]
right.start_date = first_week_in_right[0]
left.label_side = "left"
right.label_side = "right"
return left, right
def generate_calendar(outfile, start_date=None, week_start=0):
c = canvas.Canvas(outfile, pagesize=letter)
c.setAuthor("Michael Brown")
w, h = letter
center = w / 2
if start_date is None:
start_date = date.today()
cal = CalendarDrawing(start_date, week_start=week_start)
left_col, right_col = cal.split()
middle_padding = 50
left_col.draw(c, center - middle_padding/2 - left_col.get_width(), h-195, show_year_on_first_row=True)
right_col.draw(c, center + middle_padding/2 , h-195)
# Title
c.setFont("Helvetica", 30)
c.setFillColor("black")
c.drawCentredString(w/2, h-50, "Don't Break The Chain")
c.drawCentredString(w/2, h-100, "_______________")
# Subtitle
date_format = "%b %d, %Y"
subtitle = "{} - {}".format(
left_col.start_date.strftime(date_format),
right_col.end_date.strftime(date_format),
)
c.setFont("Helvetica", 12)
c.setFillColor(COLOR_GRAY)
c.drawCentredString(w/2, h-130, subtitle)
# PDF title
c.setTitle("Don't Break the Chain {}".format(subtitle))
# Attribution
c.setFont("Helvetica", 10)
c.setFillColor(COLOR_GRAY)
c.drawCentredString(w/2 + 200, 42, "msbrown.net/chain")
# Lines for debugging alignment
"""
p = c.beginPath()
# Center
p.moveTo(w/2, 72)
p.lineTo(w/2, h-72)
# Left
p.moveTo(36, 72)
p.lineTo(36, h-72)
# Right
p.moveTo(w-36, 72)
p.lineTo(w-36, h-72)
c.drawPath(p)
"""
c.showPage()
c.save()
def main():
generate_calendar(open("calendar.pdf", 'wb'))
if __name__ == "__main__":
main()