песочница
Для целей текущей задачи куб единичной длины отображается в косой проекции с символами ASCII следующим образом:
+-----+
/ /|
+-----+ |
| | +
| |/
+-----+
+
для вершин.-
для X ребер. Длина единицы вдоль X представлена пятью-
между двумя вершинами.|
для Y ребер. Длина единицы вдоль Y представлена двумя|
между двумя вершинами./
для Z ребер. Длина единицы вдоль Z представлена одной/
между двумя вершинами.- Вершины отображаются только там, где пересекаются все три плоскости.
- Края рисуются только там, где пересекаются ровно две плоскости.
Когда грань единицы выдавливается, она смещается на единицу длины от ее исходного положения, и для каждого направления создаются четыре новых ребра (положительное и отрицательное).
Вы можете думать о выдавливании как о рисовании осей трехмерной декартовой системы координат, где каждая ось представлена в виде кубоида с поперечным сечением 1x1 и длиной n
от (0,0,0)
Выдавливается на 1 вдоль X:
+-----------------+
/ /|
+-----------------+ |
| | +
| |/
+-----------------+
задача
Учитывая три числа для осей XYZ, вытяните грани единичного куба симметрично указанным величинам и отразите результат с символами ASCII, как указано выше.
вход
x, y, z - неотрицательные числа - длины экструзии для соответствующих осей. 0 означает отсутствие экструзии. В качестве входных данных можно указать три числа, список из трех чисел, тройку, строку или все, что вам удобно.
Выход
Рисунок ASCII куба после экструзии. Ведущие и конечные пробелы разрешены.
Контрольные примеры
X Y Z
0 0 0
+-----+
/ /|
+-----+ |
| | +
| |/
+-----+
1 0 0
+-----------------+
/ /|
+-----------------+ |
| | +
| |/
+-----------------+
0 0 1
+-----+
/ /|
/ / |
/ / +
/ / /
/ / /
+-----+ /
| | /
| |/
+-----+
1 1 0
+-----+
/ /|
+-----+ |
+---| | +-----+
/ | |/ /|
+-----+ +-----+ |
| | +
| |/
+-----+ +-----+
| | +
| |/
+-----+
2 0 1
+-----+
/ /|
+-----------+ +-----------+
/ /|
+-----------+ +-----------+ |
| / /| | +
| +-----+ | |/
+---------| | +-----------+
| |/
+-----+
1 1 1
+-----+
/ /|-+
+-----+ |/|
+---| | +-----+
/ | |/ /|
+-----+-----+-----+ |
| / /| | +
| +-----+ | |/
+---| | +-----+
| |/| +
+-----+ |/
+-----+
Критерии победы
Самое короткое решение в байтах на каждом языке выигрывает. Пожалуйста, добавьте краткое описание используемого метода и ваш код.
Ответы:
JavaScript (ES6),
525 ... 475 471459 байтСохранено 13 байтов благодаря @Neil
Принимает ввод в виде массива
[X,Y,Z]
. Возвращает матрицу символов.Попробуйте онлайн!
Как?
Рисование шагов
Выход состоит из 15 сторон, нарисованных в определенном порядке.
Реализация
Функция рисованияg . Работает со следующими параметрами:
Вершины всегда нарисованы. В зависимости от значения другого параметраc края либо отрисовываются, либо стираются.
Еслиt=0 , функция рисует лицевую сторону:
Еслиt=1 , он рисует верхнюю сторону:
Еслиt=2 , он рисует правую сторону:
Координаты(x,y) и размер (w,h) каждой стороны либо имеют постоянное значение, либо зависят точно от одной из входных переменных: X , Y или Z . Кроме того, каждая сторона рисуется тогда и только тогда, когда выполняется определенное условие.
Подводя итог, сторона полностью описывается с помощью:
Поэтому нам нужно хранить следующие 10 параметров для каждой стороны:
Ниже приведены параметры 15 сторон, которые необходимо нарисовать:
We force all values into the range[0..9] by applying the following operations:
It results in 15 numbers of exactly 10 decimal digits, which are stored as 15 groups of 7 digits in base 36.
For instance, the first side is encoded as
4032070460
and stored as1uol9h8
.источник
Array(W*2+9).fill` `
saves a byte.APL (Dyalog Classic),
162161132130 bytesTry it online!
(1 + 4*cx + 2*cy + cz) mod 16
wherecx
,cy
,cz
are the number of same-value "rods" along axis x,y,z, i.e. vectors along that axis that consist of the same value: 0 0 or 1 1. we make an exception if the subarray is all-zero (or all-one - those don't matter) and we consider its number 0 instead of 28-|/+
thanks Scott Milner for spotting that some
+
s rendered as?
sисточник
Charcoal, 325 bytes
Try it online! Link is to verbose version of code. Explanation:
Input the extrusions, but premultiply them to save bytes.
If at least two of the extrusions are zero, then simply draw a cuboid of dimensions (2x+1, 2y+1, 2z+1). Otherwise:
Print the left extrusion, if any.
Print the down extrusion, if any.
Print the back extrusion, if any.
The remaining extrusions all meet at this point (which doesn't get drawn until the very end!)
Print the front extrusion, if any, taking care to erase parts of the left and down extrusions that may overlap.
Print the up extrusion, if any, taking care to erase parts of the back and left extrusions that may overlap.
Print the right extrusion, if any, taking care to erase parts of the down and back extrusions that may overlap.
Draw the joins between the latter extrusions.
источник
Charcoal,
195164144 bytesTry it online! Link is to verbose version of code. I'm posting this as a separate answer as it uses a completely different approach to drawing the extrusion. Explanation:
Input the extrusions and calculate half the size of the enclosing cuboid, but in integer arithmetic because Charcoal's ranges are always integers. The origin of the output maps to the centre of the original unit cube.
Loop over all coordinates within (including the boundary) the cuboid containing the extrusion.
Jump to the output position corresponding to those coordinates.
From the given coordinates, peek in all eight diagonal directions to determine whether the extrusion overlaps in that direction. The peeked coordinates are checked that they still lie within the cuboid, and then the number of axes in which the coordinate lies within the original cube must be greater than 1. Note that since the cube has an odd display height, the Y-axis values that are peeked are integers while the other axes use fractional coordinates.
Consider the number of directions in which the extrusion overlaps. There are five cases of interest where we want to print something, as in the case of zero, that means that this is empty space and we don't want to print anything, while in the case of eight, that means that this is inside the extrusion and anything we did print would be overprinted by a layer nearer the eye point.
If the extrusion only overlaps in one direction then this is an outer corner and we need to output a
+
.If the extrusion overlaps in two directions then this is an outer edge. Which sort of edge is determined from the separation between the two overlaps; 6 and 7 are rear-facing edges and will get overwritten, 4 is a diagonal edge, 2 is a vertical edge and 1 is a horizontal edge. (I actually calculate 7 minus the separation as it seems to be easier to do.)
If the extrusion overlaps in three directions then this is an inner corner in the case where one of the extrusions is zero and we need to output a
+
.If the extrusion overlaps in four directions then there are two cases: faces (any direction), and inner corners in the case with three positive extrusions. In the latter case there are an odd number of overlaps towards the viewer.
If the extrusion overlaps in six directions then this is an inner edge. It works like the complement of an outer edge except that we're only interested when one of the two empty spaces is the direction towards the eye point (the last entry in the array).
источник
K (ngn/k), 172 bytes
Try it online!
obligatory k rewrite of my apl solution
same algorithm, except that 3d->2d rendering is done with (the k equivalent of) scatter index assignment instead of creating 2d matrices for each 3d element and mixing them
источник
ngn/apl
perfrom in comparison to yourDyalog APL
solution?⍤
) and stencil (⌺
)