forked from pencelheimer/typst_nure_template
usable
This commit is contained in:
1
.gitignore
vendored
1
.gitignore
vendored
@ -0,0 +1 @@
|
|||||||
|
*.pdf
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Typst template for NURE works.
|
# Typst template for NURE works.
|
||||||
|
|
||||||
> [!WARNING]
|
> [!WARNING]
|
||||||
> Extremely WIP!
|
> Still WIP, but you can use it.
|
||||||
|
80
bibl.yml
Normal file
80
bibl.yml
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
go:
|
||||||
|
type: Web
|
||||||
|
title: The Go Programming Language
|
||||||
|
author: The Go Programming Language
|
||||||
|
url:
|
||||||
|
value: https://go.dev/
|
||||||
|
date: 2024-12-10
|
||||||
|
|
||||||
|
|
||||||
|
htmx:
|
||||||
|
type: Web
|
||||||
|
title: Htmx - high power tools for html
|
||||||
|
author: Htmx - high power tools for html
|
||||||
|
url:
|
||||||
|
value: https://htmx.org/
|
||||||
|
date: 2024-12-10
|
||||||
|
|
||||||
|
mysql:
|
||||||
|
type: Book
|
||||||
|
title: MySQL Language Reference
|
||||||
|
author: Ab M.
|
||||||
|
publisher: MySQL Press
|
||||||
|
date: 2004
|
||||||
|
page-total: 600
|
||||||
|
|
||||||
|
neovim:
|
||||||
|
type: Web
|
||||||
|
title: About neovim
|
||||||
|
author: Neovim
|
||||||
|
url:
|
||||||
|
value: https://neovim.io/charter/
|
||||||
|
date: 2024-12-10
|
||||||
|
|
||||||
|
echo:
|
||||||
|
type: Web
|
||||||
|
title: High performance, extensible, minimalist Go web framework | Echo
|
||||||
|
author: High performance, extensible, minimalist Go web framework | Echo
|
||||||
|
url:
|
||||||
|
value: https://echo.labstack.com/
|
||||||
|
date: 2024-12-10
|
||||||
|
|
||||||
|
sqlx:
|
||||||
|
type: Web
|
||||||
|
title: Illustrated Guide to SQLX
|
||||||
|
author: GitHub Pages
|
||||||
|
url:
|
||||||
|
value: http://jmoiron.github.io/sqlx/
|
||||||
|
date: 2024-12-10
|
||||||
|
|
||||||
|
human:
|
||||||
|
type: Web
|
||||||
|
title: Про нас | HUMAN
|
||||||
|
author: HUMAN
|
||||||
|
url:
|
||||||
|
value: https://human.ua/
|
||||||
|
date: 2024-12-11
|
||||||
|
|
||||||
|
docker:
|
||||||
|
type: Web
|
||||||
|
title: "Docker: accelerated container application development"
|
||||||
|
author: Docker
|
||||||
|
url:
|
||||||
|
value: https://www.docker.com/
|
||||||
|
date: 2024-12-15
|
||||||
|
|
||||||
|
docker_compose:
|
||||||
|
type: Web
|
||||||
|
title: Docker compose
|
||||||
|
author: Docker Documentation
|
||||||
|
url:
|
||||||
|
value: https://docs.docker.com/compose/
|
||||||
|
date: 2024-12-15
|
||||||
|
|
||||||
|
hateoas:
|
||||||
|
type: Web
|
||||||
|
title: HATEOAS and Why It's Needed in RESTful API?
|
||||||
|
author: GeeksforGeeks
|
||||||
|
url:
|
||||||
|
value: https://www.geeksforgeeks.org/hateoas-and-why-its-needed-in-restful-api/
|
||||||
|
date: 2024-12-15
|
267
default.typ
267
default.typ
@ -1,267 +0,0 @@
|
|||||||
|
|
||||||
#import "@preview/indenta:0.0.3": fix-indent
|
|
||||||
|
|
||||||
#let subjects = (
|
|
||||||
БД: (
|
|
||||||
name: "Бази даних",
|
|
||||||
mentor: (
|
|
||||||
name: "Черепанова Ю. Ю.",
|
|
||||||
gender: "f",
|
|
||||||
degree: "ст. викл. каф. ПІ",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
/// DSTU 3008:2015 Template for NURE
|
|
||||||
/// -> content
|
|
||||||
/// - doc (content): Content to apply the template to.
|
|
||||||
/// - title (string): Title of the document.
|
|
||||||
/// - authors: List of Author dicts.
|
|
||||||
/// - mentor: Mentor dict.
|
|
||||||
/// - include_toc: Include table of contents?
|
|
||||||
/// - doctype ("lb" | "pz" | "ku"): Document type.
|
|
||||||
/// - worknumber: Number of the work, can be omitted.
|
|
||||||
/// - subject: Subject name.
|
|
||||||
#let conf(
|
|
||||||
doc,
|
|
||||||
title: [Work Title],
|
|
||||||
authors: ((name: "Author Name", variant: 1, group: "Group Name", gender: "m"),),
|
|
||||||
include_toc: false,
|
|
||||||
doctype: "ЛБ",
|
|
||||||
worknumber: 1,
|
|
||||||
subject_shorthand: "NONE",
|
|
||||||
) = {
|
|
||||||
// numless header
|
|
||||||
let numless(it) = {
|
|
||||||
set heading(numbering: none)
|
|
||||||
it
|
|
||||||
}
|
|
||||||
|
|
||||||
// LaTeX-like vfill shorthand
|
|
||||||
let vfill = () => v(1fr)
|
|
||||||
|
|
||||||
set document(title: title, author: authors.at(0).name)
|
|
||||||
|
|
||||||
set page(
|
|
||||||
paper: "a4",
|
|
||||||
margin: (top: 20mm, right: 10mm, bottom: 20mm, left: 25mm),
|
|
||||||
number-align: top + right,
|
|
||||||
numbering: (..numbers) => {
|
|
||||||
if numbers.pos().at(0) != 1 {
|
|
||||||
numbering("1", numbers.pos().at(0))
|
|
||||||
}
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
set text(font: "Times New Roman", size: 14pt, hyphenate: false, lang: "ua")
|
|
||||||
set par(justify: true, first-line-indent: 1.25cm)
|
|
||||||
|
|
||||||
// set 1.5 line spacing
|
|
||||||
let spacing = 0.95em
|
|
||||||
set block(spacing: spacing)
|
|
||||||
set par(spacing: spacing)
|
|
||||||
set par(leading: spacing)
|
|
||||||
|
|
||||||
// enums and lists
|
|
||||||
let ua_alph(pattern: "а)") = {
|
|
||||||
// INFO: This alphabet is not full, maybe it should be extended or maybe not.
|
|
||||||
// I cant remember nor find proper formatting rules.
|
|
||||||
// "абвгґдеєжзиіїйклмнопрстуфхцчшщьюя" (full alphabet)
|
|
||||||
|
|
||||||
let alphabet = "абвгдежиклмнпрстуфхцшщюя".split("")
|
|
||||||
|
|
||||||
i => {
|
|
||||||
let letter = alphabet.at(i)
|
|
||||||
let str = ""
|
|
||||||
for char in pattern {
|
|
||||||
if char == "а" {
|
|
||||||
str += letter
|
|
||||||
} else if char == "А" {
|
|
||||||
str += upper(letter)
|
|
||||||
} else {
|
|
||||||
str += char
|
|
||||||
}
|
|
||||||
}
|
|
||||||
str
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
set enum(numbering: ua_alph(pattern: "а)"), indent: 1.25cm, body-indent: 0.5cm)
|
|
||||||
show enum: it => {
|
|
||||||
set enum(indent: 0em, numbering: "1)")
|
|
||||||
it
|
|
||||||
}
|
|
||||||
|
|
||||||
set list(indent: 1.35cm, body-indent: 0.5cm, marker: [--])
|
|
||||||
|
|
||||||
// figures
|
|
||||||
set figure.caption(separator: [ -- ])
|
|
||||||
|
|
||||||
let fig = counter("figure")
|
|
||||||
let tab = counter("table")
|
|
||||||
|
|
||||||
show figure.where(kind: image): set figure(
|
|
||||||
numbering: (..) => {
|
|
||||||
fig.step()
|
|
||||||
context str(counter(heading).get().at(0) + worknumber - 1) + "." + context fig.display()
|
|
||||||
},
|
|
||||||
)
|
|
||||||
show figure.where(kind: table): set figure(
|
|
||||||
numbering: (..) => {
|
|
||||||
tab.step()
|
|
||||||
context str(counter(heading).get().at(0) + worknumber - 1) + "." + context tab.display()
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
// TODO: Maybe this will be better. Must be investigated.
|
|
||||||
//
|
|
||||||
// set math.equation(numbering: (..num) =>
|
|
||||||
// numbering("(1.1)", counter(heading).get().first(), num.pos().first())
|
|
||||||
// )
|
|
||||||
// set figure(numbering: (..num) =>
|
|
||||||
// numbering("1.1", counter(heading).get().first(), num.pos().first())
|
|
||||||
// )
|
|
||||||
|
|
||||||
show figure: it => {
|
|
||||||
v(spacing * 2, weak: true)
|
|
||||||
it
|
|
||||||
v(spacing * 2, weak: true)
|
|
||||||
}
|
|
||||||
|
|
||||||
// headings
|
|
||||||
// set heading(numbering: "1.1")
|
|
||||||
|
|
||||||
// HACK: I can't set initial value for headers counter, so change is only visual. If this style is changed do not forget to fix images and tables numbering as well.
|
|
||||||
set heading(numbering: (n1, ..x) => numbering("1.1", n1 - 1 + worknumber, ..x))
|
|
||||||
|
|
||||||
|
|
||||||
show heading.where(level: 1): it => {
|
|
||||||
set align(center)
|
|
||||||
set text(size: 14pt, weight: "semibold")
|
|
||||||
|
|
||||||
pagebreak(weak: true)
|
|
||||||
|
|
||||||
block(width: 100%, spacing: 0em)[ #upper(it) ]
|
|
||||||
}
|
|
||||||
show heading.where(level: 2): it => {
|
|
||||||
set text(size: 14pt, weight: "regular")
|
|
||||||
|
|
||||||
block(width: 100%, spacing: 0em)[
|
|
||||||
#h(1.25cm)
|
|
||||||
#counter(heading).display(it.numbering)
|
|
||||||
#it.body
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
let fix-spacings() = doc => {
|
|
||||||
if not doc.has("children") { return doc }
|
|
||||||
|
|
||||||
let elems = doc.children.filter(x => x != [ ] and x.func() != parbreak)
|
|
||||||
|
|
||||||
let is_heading = e => e != none and e.func() == heading
|
|
||||||
|
|
||||||
for (i, elem) in elems.enumerate() {
|
|
||||||
if not is_heading(elem) {
|
|
||||||
elem
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
let prev_elem = elems.at(i - 1, default: none)
|
|
||||||
let next_elem = elems.at(i + 1, default: none)
|
|
||||||
|
|
||||||
if elem.depth == 2 {
|
|
||||||
v(spacing * (if not is_heading(prev_elem) { 2 } else { 1 }), weak: true)
|
|
||||||
}
|
|
||||||
|
|
||||||
elem
|
|
||||||
|
|
||||||
v(spacing * (if not is_heading(next_elem) { 2 } else { 1 }), weak: true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
show: fix-spacings()
|
|
||||||
show: fix-indent()
|
|
||||||
|
|
||||||
if doctype in ("ЛБ", "ПЗ") {
|
|
||||||
let subject = subjects.at(subject_shorthand, default: "NONE")
|
|
||||||
let mentor = subject.mentor
|
|
||||||
|
|
||||||
align(
|
|
||||||
center,
|
|
||||||
[
|
|
||||||
МІНІСТЕРСТВО ОСВІТИ І НАУКИ УКРАЇНИ
|
|
||||||
|
|
||||||
ХАРКІВСЬКИЙ НАЦІОНАЛЬНИЙ УНІВЕРСИТЕТ РАДІОЕЛЕКТРОНІКИ
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
Кафедра Програмної інженерії
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
Звіт
|
|
||||||
|
|
||||||
з
|
|
||||||
#if doctype == "ЛБ" { [лабораторної роботи] } else { [практичної роботи] }
|
|
||||||
#if worknumber != none { [№ #worknumber] }
|
|
||||||
|
|
||||||
з дисципліни: "#subject.name"
|
|
||||||
|
|
||||||
з теми: "#title"
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
#columns(2)[
|
|
||||||
#set align(left)
|
|
||||||
|
|
||||||
#if authors.len() == 1 {
|
|
||||||
let author = authors.at(0)
|
|
||||||
if author.gender == "m" { [Виконав:\ ] } else { [Виконала:\ ] }
|
|
||||||
[
|
|
||||||
ст. гр. #author.group\
|
|
||||||
#author.name\
|
|
||||||
#if author.variant != none { [Варіант: №#author.variant] }
|
|
||||||
]
|
|
||||||
} else {
|
|
||||||
[
|
|
||||||
Виконали:\
|
|
||||||
ст. гр. #authors.at(0).group\
|
|
||||||
#authors.map(a => [ #a.name\ ])
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
#colbreak()
|
|
||||||
#set align(right)
|
|
||||||
|
|
||||||
#if mentor.gender == "m" { [Перевірив:\ ] } else { [Перевірила:\ ] }
|
|
||||||
#mentor.degree #if mentor.degree.len() >= 15 {
|
|
||||||
[\
|
|
||||||
]
|
|
||||||
}
|
|
||||||
#mentor.name\
|
|
||||||
]
|
|
||||||
|
|
||||||
#vfill()
|
|
||||||
|
|
||||||
Харків -- #datetime.today().display("[year]")
|
|
||||||
],
|
|
||||||
)
|
|
||||||
} else if doctype == "КУ" { }
|
|
||||||
|
|
||||||
if include_toc {
|
|
||||||
outline(
|
|
||||||
title: [
|
|
||||||
ЗМІСТ
|
|
||||||
#v(spacing * 2)
|
|
||||||
],
|
|
||||||
depth: 2,
|
|
||||||
indent: auto,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
doc
|
|
||||||
}
|
|
127
example.typ
Normal file
127
example.typ
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
#import "template.typ": *
|
||||||
|
|
||||||
|
#let authors = (
|
||||||
|
(
|
||||||
|
name: "Ситник Є. С.",
|
||||||
|
full_name_gen: "Ситника Єгора Сергійовича",
|
||||||
|
variant: 13,
|
||||||
|
group: "ПЗПІ-23-2",
|
||||||
|
gender: "m",
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
#let mentors = (
|
||||||
|
(
|
||||||
|
name: "Черепанова Ю. Ю.",
|
||||||
|
gender: "f",
|
||||||
|
degree: "Ст. викл. каф. ПІ",
|
||||||
|
),
|
||||||
|
(
|
||||||
|
name: "Русакова Н. Є.",
|
||||||
|
gender: "f",
|
||||||
|
degree: "Доц. каф. ПІ",
|
||||||
|
),
|
||||||
|
(
|
||||||
|
name: "Широкопетлєва М. С.",
|
||||||
|
gender: "f",
|
||||||
|
degree: "Ст. викл. каф. ПІ",
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
#let task_list = (
|
||||||
|
done_date: datetime(year: 2024, month: 12, day: 27),
|
||||||
|
initial_data: datetime(year: 2024, month: 9, day: 15),
|
||||||
|
source: "методичні вказівки до виконання курсової роботи, вимоги до інформаційної системи, предметна область, що пов’язана з управлінням класом та класним керівництвом.",
|
||||||
|
content: "вступ, аналіз предметної області; постановка задачі; проектування бази даних; опис програми; висновки; перелік джерел посилання.",
|
||||||
|
graphics: "загальна діаграма класів, ER-діаграма, UML-діаграми, DFD-діаграма, схема БД в 1НФ, 2НФ, 3НФ, копії екранів (“скриншоти”) прикладної програми, приклади звітів прикладної програми.",
|
||||||
|
)
|
||||||
|
|
||||||
|
#let calendar_plan = (
|
||||||
|
plan_table: [],
|
||||||
|
approval_date: datetime(year: 2024, month: 12, day: 27),
|
||||||
|
)
|
||||||
|
|
||||||
|
#let abstract = (
|
||||||
|
keywords: (
|
||||||
|
"БАЗА ДАНИХ",
|
||||||
|
"АВТОМАТИЗАЦІЯ",
|
||||||
|
"КЛАСНИЙ КЕРІВНИК",
|
||||||
|
"КЛАС",
|
||||||
|
"ШКОЛА",
|
||||||
|
"GO",
|
||||||
|
"HTMX",
|
||||||
|
"MYSQL",
|
||||||
|
"SQL",
|
||||||
|
),
|
||||||
|
text: [
|
||||||
|
Мета даної роботи -- проєктування та розробка інформаційної системи «Помічник класного керівника. Керування класом», яка спрямована на автоматизацію процесів управління класом, облік даних про учнів, планування та аналіз навчального процесу. Основна задача інформаційної системи – спростити роботу класного керівника, забезпечити ефективну організацію документації та взаємодію з учасниками освітнього процесу.
|
||||||
|
|
||||||
|
Для реалізації системи було використано сучасний стек технологій, а саме: Go -- як основна мова програмування для створення серверної логіки, HTMX -- для динамічного оновлення інтерфейсу без використання складних фреймворків, MySQL -- як СУБД для зберігання даних про учнів, їх оцінки та розклад, Neovim -- як середовище для швидкої та ефективної розробки коду, Go Echo -- веб-фреймворк для створення REST API, Go SQLx -- бібліотека для роботи з базою даних, що забезпечує зручність і гнучкість.
|
||||||
|
|
||||||
|
Результат роботи – веб-додаток, який дозволяє обліковувати особисті дані учнів та їхніх опікунів, включаючи інформацію про успішність, відвідуваність та інші показники; планувати розклад занять; генерувати звіти про успішність учнів та переглядати різну статистику. Інтерфейс, створений з використанням HTMX, легко адаптується під потреби користувача.
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
#let appendices = (
|
||||||
|
(
|
||||||
|
title: "Приклад звіту 1",
|
||||||
|
content: [test],
|
||||||
|
),
|
||||||
|
(
|
||||||
|
title: "Приклад звіту 2",
|
||||||
|
content: [test],
|
||||||
|
),
|
||||||
|
(
|
||||||
|
title: "Приклад звіту 3",
|
||||||
|
content: [test],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
#show: conf.with(
|
||||||
|
doctype: "КП",
|
||||||
|
title: "Інформаційна система «Помічник класного керівника». Керування класом",
|
||||||
|
subject_shorthand: "БД",
|
||||||
|
department_gen: "Програмної інженерії",
|
||||||
|
edu_program: "ПЗПІ",
|
||||||
|
authors: authors,
|
||||||
|
mentors: mentors,
|
||||||
|
task_list: task_list,
|
||||||
|
calendar_plan: calendar_plan,
|
||||||
|
abstract: abstract,
|
||||||
|
bib_path: "bibl.yml",
|
||||||
|
appendices: appendices,
|
||||||
|
)
|
||||||
|
|
||||||
|
#heading(depth: 1, numbering: none)[Вступ]
|
||||||
|
|
||||||
|
У сучасному освітньому середовищі інформаційні системи відіграють
|
||||||
|
ключову роль у забезпеченні ефективного управління та автоматизації процесів.
|
||||||
|
Особливо це важливо для шкіл, де необхідно систематизувати облік даних про
|
||||||
|
учнів, планувати навчальний процес і створювати звіти. Відсутність зручних
|
||||||
|
інструментів ускладнює роботу класного керівника, змушуючи його витрачати
|
||||||
|
багато часу на тяганину з документами, зокрема на складання звітів, ведення
|
||||||
|
обліку даних про учнів і підготовку розкладу, що потребує сучасних рішень для
|
||||||
|
організації взаємодії між учасниками освітнього процесу.
|
||||||
|
|
||||||
|
Метою цієї курсової роботи є розробка інформаційної системи «Помічник
|
||||||
|
класного керівника. Керування класом», яка спрощує процес управління класом,
|
||||||
|
дозволяючи автоматизувати складання звітів, зберігати та обробляти дані про
|
||||||
|
учнів, а також планувати навчальний процес у зручному форматі, забезпечує
|
||||||
|
зручний облік даних про учнів і сприяє ефективному плануванню навчального
|
||||||
|
процесу. У процесі роботи над системою було проведено аналіз потреб
|
||||||
|
користувачів, спроєктовано реляційну базу даних і розроблено веб-додаток, який
|
||||||
|
дозволяє обліковувати особисті дані учнів та їхніх опікунів, відстежувати
|
||||||
|
успішність і відвідуваність, планувати розклад занять, формувати звіти та
|
||||||
|
переглядати статистичні дані.
|
||||||
|
|
||||||
|
Основну серверну логіку написано мовою програмування Go@go, а для
|
||||||
|
забезпечення роботи інтерфейсу застосовано HTMX@htmx. Дані зберігаються у базі
|
||||||
|
даних MySQL@mysql, взаємодія з якою здійснюється через бібліотеку Go SQLx@sqlx.
|
||||||
|
Для створення REST API використано веб-фреймворк Go Echo@echo.
|
||||||
|
Розробка виконувалася у середовищі Neovim@neovim.
|
||||||
|
|
||||||
|
= Аналіз та концептуальне моделювання предметної області
|
||||||
|
#v(-1em)
|
||||||
|
== test
|
||||||
|
test
|
||||||
|
== test
|
609
template.typ
Normal file
609
template.typ
Normal file
@ -0,0 +1,609 @@
|
|||||||
|
#import "@preview/indenta:0.0.3": fix-indent
|
||||||
|
|
||||||
|
#let hfill(width) = box(width: width, repeat(" ")) // NOTE: This is a HAIR SPACE (U+200A ), not a regular space
|
||||||
|
|
||||||
|
#let uline(align: center, content) = underline([
|
||||||
|
#if align != left { hfill(1fr) }
|
||||||
|
#content
|
||||||
|
#if align != right { hfill(1fr) }
|
||||||
|
])
|
||||||
|
|
||||||
|
#let bold(content) = text(weight: "bold")[#content]
|
||||||
|
|
||||||
|
#let fig(path, caption) = [
|
||||||
|
#figure(
|
||||||
|
image(path),
|
||||||
|
caption: caption,
|
||||||
|
)
|
||||||
|
#label(path.split("/").last().split(".").first())
|
||||||
|
]
|
||||||
|
|
||||||
|
#let subjects = (
|
||||||
|
"БД": "Бази даних",
|
||||||
|
)
|
||||||
|
|
||||||
|
#let edu_programs = (
|
||||||
|
"ПЗПІ": (
|
||||||
|
name: "Інженерія програмного забезпечення",
|
||||||
|
number: 121,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
#let month_gen(month) = (
|
||||||
|
"січня",
|
||||||
|
"лютого",
|
||||||
|
"березня",
|
||||||
|
"квітня",
|
||||||
|
"травня",
|
||||||
|
"червня",
|
||||||
|
"липня",
|
||||||
|
"серпня",
|
||||||
|
"вересня",
|
||||||
|
"жовтня",
|
||||||
|
"листопада",
|
||||||
|
"грудня",
|
||||||
|
).at(month - 1)
|
||||||
|
|
||||||
|
#let spacing = 0.95em
|
||||||
|
|
||||||
|
/// DSTU 3008:2015 Template for NURE
|
||||||
|
/// -> content
|
||||||
|
/// - doc (content): Content to apply the template to.
|
||||||
|
/// - doctype ("ЛБ" | "ПЗ" | "КП"): Document type.
|
||||||
|
/// - title (str): Title of the document.
|
||||||
|
/// - subject_shorthand (str): Subject name.
|
||||||
|
/// - department_gen (str): Department name in genitive form.
|
||||||
|
/// - edu_program_number (int): Education program number.
|
||||||
|
/// - worknumber (int): Number of the work, can be omitted.
|
||||||
|
/// - authors ((name: str, full_name_gen: str, variant: int, group: str, gender: str),): List of Authors dicts.
|
||||||
|
/// - task_list (done_date: datetime, initial_data: datetime, source: str, content: str, graphics: str): Task list.
|
||||||
|
/// - calendar_plan (plan_table: list, approval_date: datetime): Calendar plan.
|
||||||
|
/// - abstract (keywords: (str,), text: [str,]): Abstract.
|
||||||
|
#let conf(
|
||||||
|
doc,
|
||||||
|
title: "NONE",
|
||||||
|
doctype: "NONE",
|
||||||
|
subject_shorthand: "NONE",
|
||||||
|
department_gen: "Програмної інженерії",
|
||||||
|
worknumber: 1,
|
||||||
|
authors: (),
|
||||||
|
mentors: (),
|
||||||
|
edu_program: "NONE",
|
||||||
|
task_list: (),
|
||||||
|
calendar_plan: (),
|
||||||
|
abstract: (),
|
||||||
|
bib_path: "bibl.yml",
|
||||||
|
appendices: (),
|
||||||
|
) = {
|
||||||
|
set document(title: title, author: authors.at(0).name)
|
||||||
|
|
||||||
|
set page(
|
||||||
|
paper: "a4",
|
||||||
|
margin: (top: 20mm, right: 10mm, bottom: 20mm, left: 25mm),
|
||||||
|
number-align: top + right,
|
||||||
|
numbering: (..numbers) => {
|
||||||
|
if numbers.pos().at(0) != 1 {
|
||||||
|
numbering("1", numbers.pos().at(0))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
set text(font: "Liberation Serif", size: 14pt, hyphenate: false, lang: "ua")
|
||||||
|
set par(justify: true, first-line-indent: 1.25cm)
|
||||||
|
set underline(evade: false)
|
||||||
|
|
||||||
|
// set 1.5 line spacing
|
||||||
|
set block(spacing: spacing)
|
||||||
|
set par(spacing: spacing)
|
||||||
|
set par(leading: spacing)
|
||||||
|
|
||||||
|
// enums and lists
|
||||||
|
let ua_alph_numbering() = {
|
||||||
|
// INFO: This alphabet is not full, maybe it should be extended or maybe not.
|
||||||
|
// I cant remember nor find proper formatting rules.
|
||||||
|
// "абвгґдеєжзиіїйклмнопрстуфхцчшщьюя" (full alphabet)
|
||||||
|
|
||||||
|
let alphabet = "абвгдежиклмнпрстуфхцшщюя".split("")
|
||||||
|
i => { alphabet.at(i) + ")" }
|
||||||
|
}
|
||||||
|
|
||||||
|
set enum(numbering: ua_alph_numbering(), indent: 1.25cm, body-indent: 0.5cm)
|
||||||
|
show enum: it => {
|
||||||
|
set enum(indent: 0em, numbering: "1)")
|
||||||
|
it
|
||||||
|
}
|
||||||
|
|
||||||
|
set list(indent: 1.35cm, body-indent: 0.5cm, marker: [--])
|
||||||
|
|
||||||
|
// citations
|
||||||
|
let bib-count = state("citation-counter", ())
|
||||||
|
show cite: it => {
|
||||||
|
it
|
||||||
|
bib-count.update(((..c)) => (..c, it.key))
|
||||||
|
}
|
||||||
|
let cit = context bib-count.final().dedup().len()
|
||||||
|
|
||||||
|
// figures
|
||||||
|
set figure.caption(separator: [ -- ])
|
||||||
|
|
||||||
|
let img = counter("image")
|
||||||
|
let tab = counter("table")
|
||||||
|
|
||||||
|
show figure.where(kind: image): set figure(
|
||||||
|
numbering: (..) => {
|
||||||
|
img.step()
|
||||||
|
context str(counter(heading).get().at(0) + worknumber - 1) + "." + context img.display()
|
||||||
|
},
|
||||||
|
)
|
||||||
|
show figure.where(kind: table): set figure(
|
||||||
|
numbering: (..) => {
|
||||||
|
tab.step()
|
||||||
|
context str(counter(heading).get().at(0) + worknumber - 1) + "." + context tab.display()
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// TODO: Maybe this will be better. Must be investigated.
|
||||||
|
//
|
||||||
|
// set math.equation(numbering: (..num) =>
|
||||||
|
// numbering("(1.1)", counter(heading).get().first(), num.pos().first())
|
||||||
|
// )
|
||||||
|
// set figure(numbering: (..num) =>
|
||||||
|
// numbering("1.1", counter(heading).get().first(), num.pos().first())
|
||||||
|
// )
|
||||||
|
|
||||||
|
show figure: it => {
|
||||||
|
v(spacing * 2, weak: true)
|
||||||
|
it
|
||||||
|
v(spacing * 2, weak: true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// headings
|
||||||
|
// HACK: I can't set initial value for headers counter, so change is only visual. If this style is changed do not forget to fix images and tables numbering as well.
|
||||||
|
set heading(numbering: (n1, ..x) => numbering("1.1", n1 - 1 + worknumber, ..x))
|
||||||
|
|
||||||
|
show heading.where(level: 1): it => {
|
||||||
|
set align(center)
|
||||||
|
set text(size: 14pt, weight: "semibold")
|
||||||
|
|
||||||
|
pagebreak(weak: true)
|
||||||
|
upper(it)
|
||||||
|
v(spacing * 2, weak: true)
|
||||||
|
}
|
||||||
|
show heading.where(level: 2): it => {
|
||||||
|
set text(size: 14pt, weight: "regular")
|
||||||
|
|
||||||
|
v(spacing * 2, weak: true)
|
||||||
|
block(width: 100%, spacing: 0em)[
|
||||||
|
#h(1.25cm)
|
||||||
|
#counter(heading).display(it.numbering)
|
||||||
|
#it.body
|
||||||
|
]
|
||||||
|
v(spacing * 2, weak: true)
|
||||||
|
}
|
||||||
|
|
||||||
|
show: fix-indent()
|
||||||
|
|
||||||
|
let subject = subjects.at(subject_shorthand, default: "NONE")
|
||||||
|
|
||||||
|
if doctype in ("ЛБ", "ПЗ") {
|
||||||
|
let mentor = subject.mentors.at(0)
|
||||||
|
align(center)
|
||||||
|
[
|
||||||
|
МІНІСТЕРСТВО ОСВІТИ І НАУКИ УКРАЇНИ
|
||||||
|
|
||||||
|
ХАРКІВСЬКИЙ НАЦІОНАЛЬНИЙ УНІВЕРСИТЕТ РАДІОЕЛЕКТРОНІКИ
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
Кафедра Програмної інженерії
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
Звіт
|
||||||
|
|
||||||
|
з
|
||||||
|
#if doctype == "ЛБ" [лабораторної роботи] else [практичної роботи]
|
||||||
|
#if worknumber != none [№ #worknumber]
|
||||||
|
|
||||||
|
з дисципліни: "#subject.name"
|
||||||
|
|
||||||
|
з теми: "#title"
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
#columns(2)[
|
||||||
|
#set align(left)
|
||||||
|
|
||||||
|
#if authors.len() == 1 [
|
||||||
|
#let author = authors.at(0)
|
||||||
|
#if author.gender == "m" { [Виконав:\ ] } else { [Виконала:\ ] }
|
||||||
|
ст. гр. #author.group\
|
||||||
|
#author.name\
|
||||||
|
#if author.variant != none { [Варіант: №#author.variant] }
|
||||||
|
] else [
|
||||||
|
Виконали:\
|
||||||
|
ст. гр. #authors.at(0).group\
|
||||||
|
#authors.map(a => [ #a.name\ ])
|
||||||
|
]
|
||||||
|
|
||||||
|
#colbreak()
|
||||||
|
#set align(right)
|
||||||
|
|
||||||
|
#if mentor.gender == "m" { [Перевірив:\ ] } else { [Перевірила:\ ] }
|
||||||
|
#mentor.degree #if mentor.degree.len() >= 15 [\ ]
|
||||||
|
#mentor.name\
|
||||||
|
]
|
||||||
|
|
||||||
|
#v(1fr)
|
||||||
|
|
||||||
|
Харків -- #datetime.today().display("[year]")
|
||||||
|
]
|
||||||
|
} else if doctype == "КП" {
|
||||||
|
let head_mentor = mentors.at(0)
|
||||||
|
let author = authors.at(0)
|
||||||
|
let edu_program = edu_programs.at(edu_program)
|
||||||
|
|
||||||
|
// page 1
|
||||||
|
align(center)[
|
||||||
|
МІНІСТЕРСТВО ОСВІТИ І НАУКИ УКРАЇНИ
|
||||||
|
|
||||||
|
ХАРКІВСЬКИЙ НАЦІОНАЛЬНИЙ УНІВЕРСИТЕТ РАДІОЕЛЕКТРОНІКИ
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
Кафедра Програмної інженерії
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
ПОЯСНЮВАЛЬНА ЗАПИСКА
|
||||||
|
|
||||||
|
ДО КУРСОВОЇ РОБОТИ
|
||||||
|
|
||||||
|
з дисципліни: "#subject"
|
||||||
|
|
||||||
|
Тема роботи: "#title"
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
#columns(2, gutter: 4cm)[
|
||||||
|
#set align(left)
|
||||||
|
|
||||||
|
#if authors.len() == 1 [
|
||||||
|
#if author.gender == "m" { [Виконав\ ] } else { [Виконала\ ] } ст. гр. #author.group
|
||||||
|
]
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
Керівник:\
|
||||||
|
#head_mentor.degree
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
Робота захищена на оцінку
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
Комісія:\
|
||||||
|
#for mentor in mentors {
|
||||||
|
[#mentor.degree\
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
#colbreak()
|
||||||
|
#set align(left)
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
#author.name
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
#head_mentor.name
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
#underline(" " * 35)
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
#linebreak()
|
||||||
|
#for mentor in mentors {
|
||||||
|
[#mentor.name\
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
#v(1fr)
|
||||||
|
|
||||||
|
Харків -- #datetime.today().display("[year]")
|
||||||
|
|
||||||
|
#pagebreak()
|
||||||
|
]
|
||||||
|
//
|
||||||
|
|
||||||
|
// page 2
|
||||||
|
{
|
||||||
|
uline([Харківський національний університет радіоелектроніки])
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
grid(
|
||||||
|
columns: (100pt, 1fr),
|
||||||
|
bold([
|
||||||
|
Кафедра
|
||||||
|
Дисципліна
|
||||||
|
Спеціальність
|
||||||
|
]),
|
||||||
|
{
|
||||||
|
uline(align: left, department_gen)
|
||||||
|
linebreak()
|
||||||
|
uline(align: left, subject)
|
||||||
|
linebreak()
|
||||||
|
uline(align: left, [#edu_program.number #edu_program.name])
|
||||||
|
},
|
||||||
|
)
|
||||||
|
grid(
|
||||||
|
columns: (1fr, 1fr, 1fr),
|
||||||
|
gutter: 0.3fr,
|
||||||
|
[#bold("Курс") #uline(2)], [#bold("Група") #uline(author.group)], [#bold("Семестр") #uline(3)],
|
||||||
|
)
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
linebreak()
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
align(center)[#bold([
|
||||||
|
ЗАВДАННЯ
|
||||||
|
|
||||||
|
на курсову роботу студента
|
||||||
|
])]
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
uline(align: left)[#emph(author.full_name_gen)]
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
bold([ #"1. "Тема роботи: ])
|
||||||
|
uline([#title.])
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
{
|
||||||
|
bold([ #"2. "Строк здачі закінченої роботи: ])
|
||||||
|
uline(task_list.done_date.display("[day].[month].[year]"))
|
||||||
|
hfill(10fr)
|
||||||
|
}
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
bold([ #"3. "Вихідні дані для роботи: ])
|
||||||
|
uline(task_list.source)
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
bold([ #"4. "Зміст розрахунково-пояснювальної записки: ])
|
||||||
|
uline(task_list.content)
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
bold([ #"5. "Перелік графічного матеріалу: ])
|
||||||
|
uline(task_list.graphics)
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
{
|
||||||
|
bold([ #"6. "Строк здачі закінченої роботи: ])
|
||||||
|
uline(task_list.done_date.display("[day].[month].[year]"))
|
||||||
|
hfill(10fr)
|
||||||
|
}
|
||||||
|
|
||||||
|
pagebreak()
|
||||||
|
}
|
||||||
|
|
||||||
|
// page 3
|
||||||
|
{
|
||||||
|
align(center)[#bold([КАЛЕНДАРНИЙ ПЛАН])]
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
calendar_plan.plan_table
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
grid(
|
||||||
|
columns: (6fr, 5fr),
|
||||||
|
grid(
|
||||||
|
columns: (1fr, 3fr, 1fr),
|
||||||
|
gutter: 0.2fr,
|
||||||
|
[
|
||||||
|
Студент
|
||||||
|
#linebreak()
|
||||||
|
Керівник
|
||||||
|
#linebreak()
|
||||||
|
#align(center, ["#calendar_plan.approval_date.day()"])
|
||||||
|
],
|
||||||
|
[
|
||||||
|
#uline(align: center, [])
|
||||||
|
#linebreak()
|
||||||
|
#uline(align: center, [])
|
||||||
|
#linebreak()
|
||||||
|
#uline(align: center, month_gen(calendar_plan.approval_date.month()))
|
||||||
|
],
|
||||||
|
{
|
||||||
|
linebreak()
|
||||||
|
linebreak()
|
||||||
|
[#calendar_plan.approval_date.year() р.]
|
||||||
|
},
|
||||||
|
),
|
||||||
|
[
|
||||||
|
#author.name,
|
||||||
|
#linebreak()
|
||||||
|
#head_mentor.degree
|
||||||
|
#head_mentor.name.
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
pagebreak()
|
||||||
|
}
|
||||||
|
|
||||||
|
// page 4 {{{
|
||||||
|
[
|
||||||
|
#align(center)[#bold([РЕФЕРАТ])]
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
#context [
|
||||||
|
#let pages = counter(page).final().at(0)
|
||||||
|
#let tables = counter("table").final().at(0)
|
||||||
|
#let images = counter("image").final().at(0)
|
||||||
|
#let bibs = bib-count.final().dedup().len()
|
||||||
|
|
||||||
|
#let counters = ()
|
||||||
|
|
||||||
|
#if pages != 0 { counters.push([#pages с.]) }
|
||||||
|
#if tables != 0 { counters.push([#tables табл.]) }
|
||||||
|
#if images != 0 { counters.push([#images рис.]) }
|
||||||
|
#if bibs != 0 { counters.push([#bibs джерел]) }
|
||||||
|
|
||||||
|
Пояснювальна записка до курсової роботи: #counters.join(", ").
|
||||||
|
]
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
#{
|
||||||
|
let keywords = abstract.keywords
|
||||||
|
let is_cyrillic = word => word
|
||||||
|
.split("")
|
||||||
|
.any(char => ("А" <= char and char <= "я") or char == "Ё" or char == "ё")
|
||||||
|
|
||||||
|
let n = keywords.len()
|
||||||
|
for i in range(n) {
|
||||||
|
for j in range(0, n - i - 1) {
|
||||||
|
if (
|
||||||
|
(not is_cyrillic(keywords.at(j)) and is_cyrillic(keywords.at(j + 1)))
|
||||||
|
or (
|
||||||
|
is_cyrillic(keywords.at(j)) == is_cyrillic(keywords.at(j + 1)) and keywords.at(j) > keywords.at(j + 1)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
(keywords.at(j), keywords.at(j + 1)) = (keywords.at(j + 1), keywords.at(j))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
keywords.join(", ")
|
||||||
|
}
|
||||||
|
|
||||||
|
#linebreak()
|
||||||
|
|
||||||
|
#abstract.text
|
||||||
|
]
|
||||||
|
// }}}
|
||||||
|
|
||||||
|
// page 5 {{{
|
||||||
|
show outline.entry: it => {
|
||||||
|
show linebreak: none
|
||||||
|
it
|
||||||
|
}
|
||||||
|
outline(
|
||||||
|
title: [
|
||||||
|
ЗМІСТ
|
||||||
|
#v(spacing * 2)
|
||||||
|
],
|
||||||
|
depth: 2,
|
||||||
|
indent: auto,
|
||||||
|
)
|
||||||
|
// }}}
|
||||||
|
}
|
||||||
|
|
||||||
|
doc
|
||||||
|
|
||||||
|
// bibliography {{{
|
||||||
|
show bibliography: it => {
|
||||||
|
set text(size: 0pt)
|
||||||
|
it
|
||||||
|
}
|
||||||
|
|
||||||
|
heading(depth: 1, numbering: none)[Перелік джерел посилання]
|
||||||
|
|
||||||
|
bibliography(
|
||||||
|
bib_path,
|
||||||
|
style: "ieee",
|
||||||
|
full: true,
|
||||||
|
title: none,
|
||||||
|
)
|
||||||
|
|
||||||
|
let bib_data = yaml(bib_path)
|
||||||
|
|
||||||
|
let format-entry(citation) = {
|
||||||
|
if (citation.type == "Web") {
|
||||||
|
let date_array = citation.url.date.split("-")
|
||||||
|
let date = datetime(
|
||||||
|
year: int(date_array.at(0)),
|
||||||
|
month: int(date_array.at(1)),
|
||||||
|
day: int(date_array.at(2)),
|
||||||
|
)
|
||||||
|
[
|
||||||
|
#citation.title.
|
||||||
|
_#{citation.author}_.
|
||||||
|
URL: #citation.url.value (дата звернення: #date.display("[day].[month].[year]")).
|
||||||
|
]
|
||||||
|
} else if citation.type == "Book" [
|
||||||
|
#citation.author
|
||||||
|
#citation.title.
|
||||||
|
#citation.publisher,
|
||||||
|
#citation.date.
|
||||||
|
#citation.page-total c.
|
||||||
|
] else [
|
||||||
|
UNSUPPORTED BIBLIOGRAPHY ENTRY TYPE, PLEASE OPEN THE ISSUE
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
show enum.item: it => {
|
||||||
|
box(width: 1.25cm)
|
||||||
|
box(width: 1em + 0.5cm)[#it.number.]
|
||||||
|
it.body
|
||||||
|
linebreak()
|
||||||
|
}
|
||||||
|
|
||||||
|
context {
|
||||||
|
let citations = query(ref.where(element: none)).map(r => str(r.target)).dedup()
|
||||||
|
|
||||||
|
for (i, citation) in citations.enumerate() {
|
||||||
|
enum.item(
|
||||||
|
i + 1,
|
||||||
|
format-entry(bib_data.at(citation)),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// }}}
|
||||||
|
|
||||||
|
// appendices {{{
|
||||||
|
counter(heading).update(0)
|
||||||
|
|
||||||
|
for (i, appendix) in appendices.enumerate() [
|
||||||
|
#set heading(
|
||||||
|
numbering: i => [
|
||||||
|
Додаток #"АБВГДЕЖИКЛМНПРСТУФХЦШЩЮЯ".split("").at(i)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
#show heading: it => {
|
||||||
|
set align(center)
|
||||||
|
set text(size: 14pt, weight: "regular")
|
||||||
|
pagebreak(weak: true)
|
||||||
|
|
||||||
|
bold(upper(counter(heading).display(it.numbering)))
|
||||||
|
|
||||||
|
linebreak()
|
||||||
|
|
||||||
|
it.body
|
||||||
|
|
||||||
|
v(spacing * 2, weak: true)
|
||||||
|
}
|
||||||
|
= #appendix.title
|
||||||
|
#appendix.content
|
||||||
|
]
|
||||||
|
// }}}
|
||||||
|
}
|
600
test.typ
600
test.typ
@ -1,600 +0,0 @@
|
|||||||
|
|
||||||
#import "@preview/indenta:0.0.3": fix-indent
|
|
||||||
|
|
||||||
#let subjects = (
|
|
||||||
"БД": (
|
|
||||||
name: "Бази даних",
|
|
||||||
mentors: (
|
|
||||||
(
|
|
||||||
name: "Черепанова Ю. Ю.",
|
|
||||||
gender: "f",
|
|
||||||
degree: "Ст. викл. каф. ПІ",
|
|
||||||
),
|
|
||||||
(
|
|
||||||
name: "Русакова Н. Є.",
|
|
||||||
gender: "f",
|
|
||||||
degree: "Доц. каф. ПІ",
|
|
||||||
),
|
|
||||||
(
|
|
||||||
name: "Широкопетлєва М. С.",
|
|
||||||
gender: "f",
|
|
||||||
degree: "Ст. викл. каф. ПІ",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
/// DSTU 3008:2015 Template for NURE
|
|
||||||
/// -> content
|
|
||||||
/// - doc (content): Content to apply the template to.
|
|
||||||
/// - title (string): Title of the document.
|
|
||||||
/// - authors: List of Author dicts.
|
|
||||||
/// - mentor: Mentor dict.
|
|
||||||
/// - include_toc: Include table of contents?
|
|
||||||
/// - doctype ("lb" | "pz" | "ku"): Document type.
|
|
||||||
/// - worknumber: Number of the work, can be omitted.
|
|
||||||
/// - subject: Subject name.
|
|
||||||
#let conf(
|
|
||||||
doc,
|
|
||||||
authors: (
|
|
||||||
(
|
|
||||||
name: "Ситник Є. С.",
|
|
||||||
full_name_gen: "Ситника Єгора Сергійовича",
|
|
||||||
variant: 14,
|
|
||||||
group: "ПЗПІ-23-2",
|
|
||||||
)
|
|
||||||
),
|
|
||||||
title: "Інформаційна система «Помічник класного керівника». Керування класом",
|
|
||||||
worknumber: 1,
|
|
||||||
doctype: "ЛБ",
|
|
||||||
subject_shorthand: "БД",
|
|
||||||
department: "Програмної інженерії",
|
|
||||||
edu_program: (
|
|
||||||
name: "Інженерія програмного забезпечення",
|
|
||||||
number: 121,
|
|
||||||
),
|
|
||||||
task_list: (
|
|
||||||
done_date: datetime(year: 2024, month: 12, day: 27),
|
|
||||||
initial_data: datetime(year: 2024, month: 9, day: 15),
|
|
||||||
source: "методичні вказівки до виконання курсової роботи, вимоги до інформаційної системи, предметна область, що пов’язана з управлінням класом та класним керівництвом.",
|
|
||||||
content: "вступ, аналіз предметної області; постановка задачі; проектування бази даних; опис програми; висновки; перелік джерел посилання.",
|
|
||||||
graphics: "загальна діаграма класів, ER-діаграма, UML-діаграми, DFD-діаграма, схема БД в 1НФ, 2НФ, 3НФ, копії екранів (“скриншоти”) прикладної програми, приклади звітів прикладної програми.",
|
|
||||||
),
|
|
||||||
calendar_plan: (
|
|
||||||
plan_table: [],
|
|
||||||
approval_date: datetime(year: 2024, month: 12, day: 27),
|
|
||||||
),
|
|
||||||
abstract: (
|
|
||||||
keywords: (
|
|
||||||
"БАЗА ДАНИХ",
|
|
||||||
"АВТОМАТИЗАЦІЯ",
|
|
||||||
"КЛАСНИЙ КЕРІВНИК",
|
|
||||||
"КЛАС",
|
|
||||||
"ШКОЛА",
|
|
||||||
"GO",
|
|
||||||
"HTMX",
|
|
||||||
"MYSQL",
|
|
||||||
"SQL",
|
|
||||||
),
|
|
||||||
text: [
|
|
||||||
Мета даної роботи – проєктування та розробка інформаційної системи «Помічник класного керівника. Керування класом», яка спрямована на автоматизацію процесів управління класом, облік даних про учнів, планування та аналіз навчального процесу. Основна задача інформаційної системи – спростити роботу класного керівника, забезпечити ефективну організацію документації та взаємодію з учасниками освітнього процесу.
|
|
||||||
|
|
||||||
Для реалізації системи було використано сучасний стек технологій, а саме: Go – як основна мова програмування для створення серверної логіки, HTMX – для динамічного оновлення інтерфейсу без використання складних фреймворків, MySQL – як СУБД для зберігання даних про учнів, їх оцінки та розклад, Neovim – як середовище для швидкої та ефективної розробки коду, Go Echo – веб-фреймворк для створення REST API, Go SQLx – бібліотека для роботи з базою даних, що забезпечує зручність і гнучкість.
|
|
||||||
|
|
||||||
Результат роботи – веб-додаток, який дозволяє обліковувати особисті дані учнів та їхніх опікунів, включаючи інформацію про успішність, відвідуваність та інші показники; планувати розклад занять; генерувати звіти про успішність учнів та переглядати різну статистику. Інтерфейс, створений з використанням HTMX, легко адаптується під потреби користувача.
|
|
||||||
],
|
|
||||||
),
|
|
||||||
include_toc: true,
|
|
||||||
) = {
|
|
||||||
// LaTeX-like vfill/hfill shorthands
|
|
||||||
let vfill = () => v(1fr)
|
|
||||||
let hfill(width) = box(width: width, repeat(" ")) // NOTE: This is a HAIR SPACE (U+200A ), not a regular space
|
|
||||||
|
|
||||||
set underline(evade: false)
|
|
||||||
|
|
||||||
let uline(align: center, content) = {
|
|
||||||
underline([
|
|
||||||
#if align != left { hfill(1fr) }
|
|
||||||
#content
|
|
||||||
#if align != right { hfill(1fr) }
|
|
||||||
])
|
|
||||||
}
|
|
||||||
|
|
||||||
// other shorthands
|
|
||||||
let bold(it) = text(weight: "bold")[#it]
|
|
||||||
|
|
||||||
let month_gen(month) = (
|
|
||||||
"січня",
|
|
||||||
"лютого",
|
|
||||||
"березня",
|
|
||||||
"квітня",
|
|
||||||
"травня",
|
|
||||||
"червня",
|
|
||||||
"липня",
|
|
||||||
"серпня",
|
|
||||||
"вересня",
|
|
||||||
"жовтня",
|
|
||||||
"листопада",
|
|
||||||
"грудня",
|
|
||||||
).at(month - 1)
|
|
||||||
|
|
||||||
set document(title: title, author: authors.at(0).name)
|
|
||||||
|
|
||||||
set page(
|
|
||||||
paper: "a4",
|
|
||||||
margin: (top: 20mm, right: 10mm, bottom: 20mm, left: 25mm),
|
|
||||||
number-align: top + right,
|
|
||||||
numbering: (..numbers) => {
|
|
||||||
if numbers.pos().at(0) != 1 {
|
|
||||||
numbering("1", numbers.pos().at(0))
|
|
||||||
}
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
set text(font: "Times New Roman", size: 14pt, hyphenate: false, lang: "ua")
|
|
||||||
set par(justify: true, first-line-indent: 1.25cm)
|
|
||||||
|
|
||||||
// set 1.5 line spacing
|
|
||||||
let spacing = 0.95em
|
|
||||||
set block(spacing: spacing)
|
|
||||||
set par(spacing: spacing)
|
|
||||||
set par(leading: spacing)
|
|
||||||
|
|
||||||
// enums and lists
|
|
||||||
let ua_alph(pattern: "а)") = {
|
|
||||||
// INFO: This alphabet is not full, maybe it should be extended or maybe not.
|
|
||||||
// I cant remember nor find proper formatting rules.
|
|
||||||
// "абвгґдеєжзиіїйклмнопрстуфхцчшщьюя" (full alphabet)
|
|
||||||
|
|
||||||
let alphabet = "абвгдежиклмнпрстуфхцшщюя".split("")
|
|
||||||
|
|
||||||
i => {
|
|
||||||
let letter = alphabet.at(i)
|
|
||||||
let str = ""
|
|
||||||
for char in pattern {
|
|
||||||
if char == "а" {
|
|
||||||
str += letter
|
|
||||||
} else if char == "А" {
|
|
||||||
str += upper(letter)
|
|
||||||
} else {
|
|
||||||
str += char
|
|
||||||
}
|
|
||||||
}
|
|
||||||
str
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
set enum(numbering: ua_alph(pattern: "а)"), indent: 1.25cm, body-indent: 0.5cm)
|
|
||||||
show enum: it => {
|
|
||||||
set enum(indent: 0em, numbering: "1)")
|
|
||||||
it
|
|
||||||
}
|
|
||||||
|
|
||||||
set list(indent: 1.35cm, body-indent: 0.5cm, marker: [--])
|
|
||||||
|
|
||||||
// figures
|
|
||||||
set figure.caption(separator: [ -- ])
|
|
||||||
|
|
||||||
let img = counter("image")
|
|
||||||
let tab = counter("table")
|
|
||||||
|
|
||||||
show figure.where(kind: image): set figure(
|
|
||||||
numbering: (..) => {
|
|
||||||
img.step()
|
|
||||||
context str(counter(heading).get().at(0) + worknumber - 1) + "." + context img.display()
|
|
||||||
},
|
|
||||||
)
|
|
||||||
show figure.where(kind: table): set figure(
|
|
||||||
numbering: (..) => {
|
|
||||||
tab.step()
|
|
||||||
context str(counter(heading).get().at(0) + worknumber - 1) + "." + context tab.display()
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
// TODO: Maybe this will be better. Must be investigated.
|
|
||||||
//
|
|
||||||
// set math.equation(numbering: (..num) =>
|
|
||||||
// numbering("(1.1)", counter(heading).get().first(), num.pos().first())
|
|
||||||
// )
|
|
||||||
// set figure(numbering: (..num) =>
|
|
||||||
// numbering("1.1", counter(heading).get().first(), num.pos().first())
|
|
||||||
// )
|
|
||||||
|
|
||||||
show figure: it => {
|
|
||||||
v(spacing * 2, weak: true)
|
|
||||||
it
|
|
||||||
v(spacing * 2, weak: true)
|
|
||||||
}
|
|
||||||
|
|
||||||
// headings
|
|
||||||
// set heading(numbering: "1.1")
|
|
||||||
|
|
||||||
// HACK: I can't set initial value for headers counter, so change is only visual. If this style is changed do not forget to fix images and tables numbering as well.
|
|
||||||
set heading(numbering: (n1, ..x) => numbering("1.1", n1 - 1 + worknumber, ..x))
|
|
||||||
|
|
||||||
|
|
||||||
show heading.where(level: 1): it => {
|
|
||||||
set align(center)
|
|
||||||
set text(size: 14pt, weight: "semibold")
|
|
||||||
|
|
||||||
pagebreak(weak: true)
|
|
||||||
|
|
||||||
block(width: 100%, spacing: 0em)[ #upper(it) ]
|
|
||||||
}
|
|
||||||
show heading.where(level: 2): it => {
|
|
||||||
set text(size: 14pt, weight: "regular")
|
|
||||||
|
|
||||||
block(width: 100%, spacing: 0em)[
|
|
||||||
#h(1.25cm)
|
|
||||||
#counter(heading).display(it.numbering)
|
|
||||||
#it.body
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
let fix-spacings() = doc => {
|
|
||||||
if not doc.has("children") { return doc }
|
|
||||||
|
|
||||||
let elems = doc.children.filter(x => x != [ ])
|
|
||||||
|
|
||||||
let is_heading = e => e != none and e.func() == heading
|
|
||||||
|
|
||||||
for (i, elem) in elems.enumerate() {
|
|
||||||
if not is_heading(elem) {
|
|
||||||
elem
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
let prev_elem = elems.at(i - 1, default: none)
|
|
||||||
let next_elem = elems.at(i + 1, default: none)
|
|
||||||
|
|
||||||
if elem.depth == 2 {
|
|
||||||
v(spacing * (if not is_heading(prev_elem) { 2 } else { 1 }), weak: true)
|
|
||||||
}
|
|
||||||
|
|
||||||
elem
|
|
||||||
|
|
||||||
v(spacing * (if not is_heading(next_elem) { 2 } else { 1 }), weak: true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
show: fix-spacings()
|
|
||||||
show: fix-indent()
|
|
||||||
|
|
||||||
let subject = subjects.at(subject_shorthand, default: "NONE")
|
|
||||||
|
|
||||||
if doctype in ("ЛБ", "ПЗ") {
|
|
||||||
let mentor = subject.mentors.at(0)
|
|
||||||
align(
|
|
||||||
center,
|
|
||||||
[
|
|
||||||
МІНІСТЕРСТВО ОСВІТИ І НАУКИ УКРАЇНИ
|
|
||||||
|
|
||||||
ХАРКІВСЬКИЙ НАЦІОНАЛЬНИЙ УНІВЕРСИТЕТ РАДІОЕЛЕКТРОНІКИ
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
Кафедра Програмної інженерії
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
Звіт
|
|
||||||
|
|
||||||
з
|
|
||||||
#if doctype == "ЛБ" { [лабораторної роботи] } else { [практичної роботи] }
|
|
||||||
#if worknumber != none { [№ #worknumber] }
|
|
||||||
|
|
||||||
з дисципліни: "#subject.name"
|
|
||||||
|
|
||||||
з теми: "#title"
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
#columns(2)[
|
|
||||||
#set align(left)
|
|
||||||
|
|
||||||
#if authors.len() == 1 {
|
|
||||||
let author = authors.at(0)
|
|
||||||
if author.gender == "m" { [Виконав:\ ] } else { [Виконала:\ ] }
|
|
||||||
[
|
|
||||||
ст. гр. #author.group\
|
|
||||||
#author.name\
|
|
||||||
#if author.variant != none { [Варіант: №#author.variant] }
|
|
||||||
]
|
|
||||||
} else {
|
|
||||||
[
|
|
||||||
Виконали:\
|
|
||||||
ст. гр. #authors.at(0).group\
|
|
||||||
#authors.map(a => [ #a.name\ ])
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
#colbreak()
|
|
||||||
#set align(right)
|
|
||||||
|
|
||||||
#if mentor.gender == "m" { [Перевірив:\ ] } else { [Перевірила:\ ] }
|
|
||||||
#mentor.degree #if mentor.degree.len() >= 15 {
|
|
||||||
[\
|
|
||||||
]
|
|
||||||
}
|
|
||||||
#mentor.name\
|
|
||||||
]
|
|
||||||
|
|
||||||
#vfill()
|
|
||||||
|
|
||||||
Харків -- #datetime.today().display("[year]")
|
|
||||||
],
|
|
||||||
)
|
|
||||||
} else if doctype == "КУ" {
|
|
||||||
let head_mentor = subject.mentors.at(0)
|
|
||||||
let author = authors.at(0)
|
|
||||||
|
|
||||||
// page 1 {{{
|
|
||||||
align(center)[
|
|
||||||
МІНІСТЕРСТВО ОСВІТИ І НАУКИ УКРАЇНИ
|
|
||||||
|
|
||||||
ХАРКІВСЬКИЙ НАЦІОНАЛЬНИЙ УНІВЕРСИТЕТ РАДІОЕЛЕКТРОНІКИ
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
Кафедра Програмної інженерії
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
ПОЯСНЮВАЛЬНА ЗАПИСКА
|
|
||||||
|
|
||||||
ДО КУРСОВОЇ РОБОТИ
|
|
||||||
|
|
||||||
з дисципліни: "#subject.name"
|
|
||||||
|
|
||||||
Тема роботи: "#title"
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
#columns(2, gutter: 4cm)[
|
|
||||||
#set align(left)
|
|
||||||
|
|
||||||
#if authors.len() == 1 [
|
|
||||||
#if author.gender == "m" { [Виконав\ ] } else { [Виконала\ ] } ст. гр. #author.group
|
|
||||||
]
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
Керівник:\
|
|
||||||
#head_mentor.degree
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
Робота захищена на оцінку
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
Комісія:\
|
|
||||||
#for mentor in subject.mentors {
|
|
||||||
[#mentor.degree\
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
#colbreak()
|
|
||||||
#set align(left)
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#author.name
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
#head_mentor.name
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#underline(" " * 35)
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
#linebreak()
|
|
||||||
#for mentor in subject.mentors {
|
|
||||||
[#mentor.name\
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
|
|
||||||
#vfill()
|
|
||||||
|
|
||||||
Харків -- #datetime.today().display("[year]")
|
|
||||||
|
|
||||||
#pagebreak()
|
|
||||||
]
|
|
||||||
// }}}
|
|
||||||
|
|
||||||
// page 2 {{{
|
|
||||||
uline([Харківський національний університет радіоелектроніки])
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
grid(
|
|
||||||
columns: (100pt, 1fr),
|
|
||||||
bold([
|
|
||||||
Кафедра
|
|
||||||
Дисципліна
|
|
||||||
Спеціальність
|
|
||||||
]),
|
|
||||||
{
|
|
||||||
uline(align: left, department)
|
|
||||||
linebreak()
|
|
||||||
uline(align: left, subject.name)
|
|
||||||
linebreak()
|
|
||||||
uline(align: left, [#edu_program.number #edu_program.name])
|
|
||||||
},
|
|
||||||
)
|
|
||||||
grid(
|
|
||||||
columns: (1fr, 1fr, 1fr),
|
|
||||||
gutter: 0.3fr,
|
|
||||||
[#bold("Курс") #uline(2)], [#bold("Група") #uline(author.group)], [#bold("Семестр") #uline(3)],
|
|
||||||
)
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
linebreak()
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
align(center)[#bold([
|
|
||||||
ЗАВДАННЯ
|
|
||||||
|
|
||||||
на курсову роботу студента
|
|
||||||
])]
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
uline(align: left)[#emph(author.full_name_gen)]
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
bold([ #"1. "Тема роботи: ])
|
|
||||||
uline([#title.])
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
{
|
|
||||||
bold([ #"2. "Строк здачі закінченої роботи: ])
|
|
||||||
uline(task_list.done_date.display("[day].[month].[year]"))
|
|
||||||
hfill(10fr)
|
|
||||||
}
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
bold([ #"3. "Вихідні дані для роботи: ])
|
|
||||||
uline(task_list.source)
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
bold([ #"4. "Зміст розрахунково-пояснювальної записки: ])
|
|
||||||
uline(task_list.content)
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
bold([ #"5. "Перелік графічного матеріалу: ])
|
|
||||||
uline(task_list.graphics)
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
{
|
|
||||||
bold([ #"6. "Строк здачі закінченої роботи: ])
|
|
||||||
uline(task_list.done_date.display("[day].[month].[year]"))
|
|
||||||
hfill(10fr)
|
|
||||||
}
|
|
||||||
|
|
||||||
pagebreak()
|
|
||||||
// }}}
|
|
||||||
|
|
||||||
// page 3 {{{
|
|
||||||
heading(numbering: none, depth: 1, [Календарний план])
|
|
||||||
|
|
||||||
calendar_plan.plan_table
|
|
||||||
|
|
||||||
linebreak()
|
|
||||||
|
|
||||||
grid(
|
|
||||||
columns: (6fr, 5fr),
|
|
||||||
grid(
|
|
||||||
columns: (1fr, 3fr, 1fr),
|
|
||||||
gutter: 0.2fr,
|
|
||||||
[
|
|
||||||
Студент
|
|
||||||
#linebreak()
|
|
||||||
Керівник
|
|
||||||
#linebreak()
|
|
||||||
#align(center, ["#calendar_plan.approval_date.day()"])
|
|
||||||
],
|
|
||||||
[
|
|
||||||
#uline(align: center, [])
|
|
||||||
#linebreak()
|
|
||||||
#uline(align: center, [])
|
|
||||||
#linebreak()
|
|
||||||
#uline(align: center, month_gen(calendar_plan.approval_date.month()))
|
|
||||||
],
|
|
||||||
{
|
|
||||||
linebreak()
|
|
||||||
linebreak()
|
|
||||||
[#calendar_plan.approval_date.year() р.]
|
|
||||||
},
|
|
||||||
),
|
|
||||||
[
|
|
||||||
#author.name,
|
|
||||||
#linebreak()
|
|
||||||
#head_mentor.degree
|
|
||||||
#head_mentor.name.
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
pagebreak()
|
|
||||||
|
|
||||||
// }}}
|
|
||||||
|
|
||||||
// page 4 {{{
|
|
||||||
[
|
|
||||||
#heading(numbering: none, depth: 1, [Реферат])
|
|
||||||
|
|
||||||
Пояснювальна записка до курсової роботи:
|
|
||||||
#context [ #counter(page).final().at(0) ] с.,
|
|
||||||
#context [ #counter("table").final().at(0) ] табл.,
|
|
||||||
#context [ #counter("image").final().at(0) ] рис.,
|
|
||||||
#context [ #counter(bibliography).final().at(0) ] джерел.
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
#{
|
|
||||||
let keywords = abstract.keywords
|
|
||||||
let is_cyrillic = word => word
|
|
||||||
.split("")
|
|
||||||
.any(char => ("А" <= char and char <= "я") or char == "Ё" or char == "ё")
|
|
||||||
|
|
||||||
let n = keywords.len()
|
|
||||||
for i in range(n) {
|
|
||||||
for j in range(0, n - i - 1) {
|
|
||||||
if (
|
|
||||||
(not is_cyrillic(keywords.at(j)) and is_cyrillic(keywords.at(j + 1)))
|
|
||||||
or (
|
|
||||||
is_cyrillic(keywords.at(j)) == is_cyrillic(keywords.at(j + 1)) and keywords.at(j) > keywords.at(j + 1)
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
(keywords.at(j), keywords.at(j + 1)) = (keywords.at(j + 1), keywords.at(j))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
keywords.join(", ")
|
|
||||||
}
|
|
||||||
|
|
||||||
#linebreak()
|
|
||||||
|
|
||||||
#abstract.text
|
|
||||||
]
|
|
||||||
// }}}
|
|
||||||
}
|
|
||||||
|
|
||||||
if include_toc {
|
|
||||||
outline(
|
|
||||||
title: [
|
|
||||||
ЗМІСТ
|
|
||||||
#v(spacing * 2)
|
|
||||||
],
|
|
||||||
depth: 2,
|
|
||||||
indent: auto,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
doc
|
|
||||||
}
|
|
||||||
|
|
||||||
#show: conf.with(
|
|
||||||
authors: (
|
|
||||||
(name: "Ситник Є. С.", full_name_gen: "Ситника Єгора Сергійовича", variant: 15, group: "ПЗПІ-23-2", gender: "m"),
|
|
||||||
),
|
|
||||||
title: "Інформаційна система «Помічник класного керівника». Керування класом",
|
|
||||||
include_toc: true,
|
|
||||||
doctype: "КУ",
|
|
||||||
subject_shorthand: "БД",
|
|
||||||
)
|
|
||||||
|
|
Reference in New Issue
Block a user