mirror of
https://github.com/JannisX11/blockbench.git
synced 2025-01-18 15:26:19 +08:00
1030 lines
30 KiB
JavaScript
1030 lines
30 KiB
JavaScript
const Painter = {
|
|
currentPixel: [-1, -1],
|
|
brushChanges: false,
|
|
current: {/*texture, image*/},
|
|
selection: {},
|
|
mirror_painting: false,
|
|
lock_alpha: false,
|
|
edit(texture, cb, options) {
|
|
if (!options.no_undo) {
|
|
Undo.initEdit({textures: [texture], bitmap: true})
|
|
}
|
|
if (texture.mode === 'link') {
|
|
texture.source = 'data:image/png;base64,' + texture.getBase64()
|
|
texture.mode = 'bitmap'
|
|
texture.saved = false
|
|
}
|
|
var instance = Painter.current[options.method === 'jimp' ? 'image' : 'canvas']
|
|
Painter.current[options.method === 'jimp' ? 'canvas' : 'image'] = undefined
|
|
|
|
var edit_name = options.no_undo ? null : (options.edit_name || 'edit texture');
|
|
|
|
if (options.use_cache &&
|
|
texture === Painter.current.texture &&
|
|
typeof instance === 'object'
|
|
) {
|
|
//IS CACHED
|
|
if (options.method === 'jimp') {
|
|
instance = cb(instance) || instance
|
|
} else {
|
|
instance = cb(instance) || instance
|
|
}
|
|
if (options.no_update === true) {
|
|
return;
|
|
}
|
|
|
|
if (options.method === 'jimp') {
|
|
Painter.current.image.getBase64(Jimp.MIME_PNG, function(a, dataUrl){
|
|
texture.updateSource(dataUrl)
|
|
if (!options.no_undo) {
|
|
Undo.finishEdit(edit_name)
|
|
}
|
|
})
|
|
} else {
|
|
texture.updateSource(instance.toDataURL())
|
|
if (!options.no_undo) {
|
|
Undo.finishEdit(edit_name)
|
|
}
|
|
}
|
|
} else {
|
|
if (options.method === 'jimp') {
|
|
Painter.current.texture = texture
|
|
Jimp.read(Buffer.from(texture.source.replace('data:image/png;base64,', ''), 'base64')).then(function(image) {
|
|
image = cb(image) || image
|
|
Painter.current.image = image
|
|
image.getBase64(Jimp.MIME_PNG, function(a, dataUrl){
|
|
texture.updateSource(dataUrl)
|
|
if (!options.no_undo) {
|
|
Undo.finishEdit(edit_name)
|
|
}
|
|
})
|
|
})
|
|
} else {
|
|
Painter.current.texture = texture
|
|
var c = Painter.current.canvas = Painter.getCanvas(texture)
|
|
c = cb(c) || c;
|
|
|
|
texture.updateSource(c.toDataURL())
|
|
if (!options.no_undo) {
|
|
Undo.finishEdit(edit_name)
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
//alpha
|
|
setAlphaMatrix(texture, x, y, val) {
|
|
if (!Painter.current.alpha_matrix) Painter.current.alpha_matrix = {}
|
|
var mx = Painter.current.alpha_matrix;
|
|
if (!mx[texture.uuid]) mx[texture.uuid] = {};
|
|
if (!mx[texture.uuid][x]) mx[texture.uuid][x] = {};
|
|
if (mx[texture.uuid][x][y]) {
|
|
val = Math.max(val, mx[texture.uuid][x][y])
|
|
}
|
|
mx[texture.uuid][x][y] = val
|
|
},
|
|
getAlphaMatrix(texture, x, y) {
|
|
return Painter.current.alpha_matrix
|
|
&& Painter.current.alpha_matrix[texture.uuid]
|
|
&& Painter.current.alpha_matrix[texture.uuid][x]
|
|
&& Painter.current.alpha_matrix[texture.uuid][x][y];
|
|
},
|
|
// Preview Brush
|
|
startPaintToolCanvas(data, e) {
|
|
if (!data && Toolbox.selected.id == 'color_picker') {
|
|
var preview = quad_previews.current
|
|
if (preview && preview.background && preview.background.imgtag) {
|
|
|
|
var ctx = Painter.getCanvas(preview.background.imgtag).getContext('2d')
|
|
var pixel_ratio = main_preview.background.imgtag.width / main_preview.background.size;
|
|
var x = (event.offsetX - preview.width/2 - preview.background.x) * pixel_ratio + main_preview.background.imgtag.width/2
|
|
var y = (event.offsetY - preview.height/2 - preview.background.y)* pixel_ratio
|
|
if (x >= 0 && y >= 0 && x < main_preview.background.imgtag.width && y < main_preview.background.imgtag.height) {
|
|
Painter.scanCanvas(ctx, x, y, 1, 1, (x, y, px) => {
|
|
var t = tinycolor({
|
|
r: px[0],
|
|
g: px[1],
|
|
b: px[2],
|
|
a: px[3]/256
|
|
})
|
|
ColorPanel.set(t)
|
|
})
|
|
}
|
|
}
|
|
}
|
|
if (!data || (data.cube && data.cube.locked)) return;
|
|
var texture = data.cube.faces[data.face].getTexture()
|
|
if (!texture || (texture.error && texture.error !== 2)) {
|
|
Blockbench.showQuickMessage('message.untextured')
|
|
return;
|
|
}
|
|
let offset = BarItems.slider_brush_size.get()%2 == 0 && Toolbox.selected.brushTool ? 0.5 : 0;
|
|
var x = Math.floor( data.intersects[0].uv.x * texture.img.naturalWidth + offset )
|
|
var y = Math.floor( (1-data.intersects[0].uv.y) * texture.img.naturalHeight + offset )
|
|
Painter.startPaintTool(texture, x, y, data.cube.faces[data.face].uv, e, data)
|
|
|
|
if (Toolbox.selected.id !== 'color_picker') {
|
|
addEventListeners(document, 'mousemove touchmove', Painter.movePaintToolCanvas, false );
|
|
addEventListeners(document, 'mouseup touchend', Painter.stopPaintToolCanvas, false );
|
|
}
|
|
},
|
|
movePaintToolCanvas(event) {
|
|
convertTouchEvent(event);
|
|
var data = Canvas.raycast(event)
|
|
if (data && data.cube && !data.cube.locked) {
|
|
var texture = data.cube.faces[data.face].getTexture()
|
|
if (texture) {
|
|
var x, y, new_face;
|
|
let offset = BarItems.slider_brush_size.get()%2 == 0 && Toolbox.selected.brushTool ? 0.5 : 0;
|
|
x = Math.floor( data.intersects[0].uv.x * texture.img.naturalWidth + offset );
|
|
y = Math.floor( (1-data.intersects[0].uv.y) * texture.img.naturalHeight + offset );
|
|
if (texture.img.naturalWidth + texture.img.naturalHeight == 0) return;
|
|
|
|
if (x === Painter.current.x && y === Painter.current.y) {
|
|
return
|
|
}
|
|
if (Painter.current.face !== data.face || Painter.current.cube !== data.cube) {
|
|
if (Toolbox.selected.id === 'draw_shape_tool') {
|
|
return;
|
|
}
|
|
Painter.current.x = x
|
|
Painter.current.y = y
|
|
Painter.current.face = data.face
|
|
Painter.current.cube = data.cube
|
|
new_face = true
|
|
if (texture !== Painter.current.texture) {
|
|
Undo.current_save.addTexture(texture)
|
|
}
|
|
}
|
|
Painter.movePaintTool(texture, x, y, event, new_face, data.cube.faces[data.face].uv)
|
|
}
|
|
}
|
|
},
|
|
stopPaintToolCanvas() {
|
|
removeEventListeners(document, 'mousemove touchmove', Painter.movePaintToolCanvas, false );
|
|
removeEventListeners(document, 'mouseup touchend', Painter.stopPaintToolCanvas, false );
|
|
Painter.stopPaintTool();
|
|
},
|
|
// Paint Tool Main
|
|
startPaintTool(texture, x, y, uvTag, event, data) {
|
|
//Called directly by startPaintToolCanvas and startBrushUV
|
|
if (Toolbox.selected.id === 'color_picker') {
|
|
Painter.colorPicker(texture, x, y)
|
|
} else if (Toolbox.selected.id === 'draw_shape_tool') {
|
|
|
|
Undo.initEdit({textures: [texture], selected_texture: true, bitmap: true});
|
|
Painter.brushChanges = false;
|
|
Painter.painting = true;
|
|
Painter.current = {
|
|
cube: data && data.cube,
|
|
face: data && data.face,
|
|
x, y,
|
|
clear: document.createElement('canvas'),
|
|
}
|
|
Painter.startPixel = [x, y];
|
|
Painter.current.clear.width = texture.width;
|
|
Painter.current.clear.height = texture.height;
|
|
Painter.current.clear.getContext('2d').drawImage(texture.img, 0, 0);
|
|
|
|
} else {
|
|
Undo.initEdit({textures: [texture], selected_texture: true, bitmap: true});
|
|
Painter.brushChanges = false;
|
|
Painter.painting = true;
|
|
|
|
if (data) {
|
|
var is_line = event.shiftKey && Painter.current.cube == data.cube && Painter.current.face == data.face
|
|
Painter.current.cube = data.cube;
|
|
Painter.current.face = data.face;
|
|
} else {
|
|
//uv editor
|
|
var is_line = event.shiftKey;
|
|
}
|
|
|
|
if (is_line) {
|
|
Painter.drawBrushLine(texture, x, y, event, false, uvTag);
|
|
} else {
|
|
Painter.current.x = Painter.current.y = 0
|
|
Painter.useBrushlike(texture, x, y, event, uvTag)
|
|
}
|
|
Painter.current.x = x;
|
|
Painter.current.y = y;
|
|
}
|
|
},
|
|
movePaintTool(texture, x, y, event, new_face, uv) {
|
|
// Called directly from movePaintToolCanvas and moveBrushUV
|
|
if (Toolbox.selected.id === 'draw_shape_tool') {
|
|
|
|
Painter.useShapeTool(texture, x, y, event, uv)
|
|
|
|
} else {
|
|
Painter.drawBrushLine(texture, x, y, event, new_face, uv)
|
|
}
|
|
Painter.current.x = x;
|
|
Painter.current.y = y;
|
|
},
|
|
stopPaintTool() {
|
|
//Called directly by stopPaintToolCanvas and stopBrushUV
|
|
if (Painter.brushChanges) {
|
|
Undo.finishEdit('paint');
|
|
Painter.brushChanges = false;
|
|
}
|
|
delete Painter.current.alpha_matrix;
|
|
Painter.painting = false;
|
|
Painter.currentPixel = [-1, -1];
|
|
},
|
|
// Tools
|
|
useBrushlike(texture, x, y, event, uvTag, no_update, is_opposite) {
|
|
if (Painter.currentPixel[0] === x && Painter.currentPixel[1] === y) return;
|
|
Painter.currentPixel = [x, y]
|
|
Painter.brushChanges = true;
|
|
let uvFactorX = 1 / Project.texture_width * texture.width;
|
|
let uvFactorY = 1 / Project.texture_height * texture.display_height;
|
|
|
|
if (Painter.mirror_painting && !is_opposite) {
|
|
Painter.runMirrorBrush(texture, x, y, event, uvTag);
|
|
}
|
|
|
|
texture.edit(function(canvas) {
|
|
var ctx = canvas.getContext('2d')
|
|
ctx.save()
|
|
|
|
ctx.beginPath();
|
|
if (uvTag) {
|
|
let anim_offset = texture.display_height * texture.currentFrame;
|
|
var rect = Painter.editing_area = [
|
|
uvTag[0] * uvFactorX,
|
|
uvTag[1] * uvFactorY + anim_offset,
|
|
uvTag[2] * uvFactorX,
|
|
uvTag[3] * uvFactorY + anim_offset
|
|
]
|
|
} else {
|
|
var rect = Painter.editing_area = [0, 0, texture.img.naturalWidth, texture.img.naturalHeight]
|
|
}
|
|
for (var t = 0; t < 2; t++) {
|
|
if (rect[t] > rect[t+2]) {
|
|
[rect[t], rect[t+2]] = [rect[t+2], rect[t]]
|
|
}
|
|
rect[t] = Math.round(rect[t])
|
|
rect[t+2] = Math.round(rect[t+2])
|
|
}
|
|
var [w, h] = [rect[2] - rect[0], rect[3] - rect[1]]
|
|
ctx.rect(rect[0], rect[1], w, h)
|
|
|
|
if (Toolbox.selected.id === 'fill_tool') {
|
|
Painter.useFilltool(texture, ctx, x, y, { rect, uvFactorX, uvFactorY, w, h })
|
|
} else {
|
|
Painter.useBrush(texture, ctx, x, y, event)
|
|
}
|
|
Painter.editing_area = undefined;
|
|
|
|
|
|
}, {no_undo: true, use_cache: true, no_update});
|
|
},
|
|
useBrush(texture, ctx, x, y, event) {
|
|
|
|
var color = tinycolor(ColorPanel.get()).toRgb();
|
|
var size = BarItems.slider_brush_size.get();
|
|
let softness = BarItems.slider_brush_softness.get()/100;
|
|
let b_opacity = BarItems.slider_brush_opacity.get()/100;
|
|
let tool = Toolbox.selected.id;
|
|
|
|
ctx.clip()
|
|
if (event.touches && event.touches[0] && event.touches[0].touchType == 'stylus' && event.touches[0].force) {
|
|
|
|
// Stylus
|
|
var touch = event.touches[0];
|
|
if (touch.force == 1) touch.force == Painter.current.force || 0;
|
|
Painter.current.force = touch.force;
|
|
|
|
if (settings.brush_opacity_modifier.value == 'pressure' && touch.force) {
|
|
b_opacity = Math.clamp(b_opacity * touch.force*1.25, 0, 100);
|
|
|
|
} else if (settings.brush_opacity_modifier.value == 'tilt' && touch.altitudeAngle !== undefined) {
|
|
var modifier = Math.clamp(1.5 / (touch.altitudeAngle + 0.3), 1, 4)/2;
|
|
b_opacity = Math.clamp(b_opacity * modifier, 0, 100);
|
|
}
|
|
if (settings.brush_size_modifier.value == 'pressure' && touch.force) {
|
|
size = Math.clamp(touch.force * size * 2, 1, 20);
|
|
|
|
} else if (settings.brush_size_modifier.value == 'tilt' && touch.altitudeAngle !== undefined) {
|
|
size *= Math.clamp(1.5 / (touch.altitudeAngle + 0.3), 1, 4);
|
|
}
|
|
}
|
|
|
|
if (tool === 'brush_tool') {
|
|
Painter.editCircle(ctx, x, y, size, softness, function(pxcolor, opacity, px, py) {
|
|
var a = b_opacity * opacity;
|
|
var before = Painter.getAlphaMatrix(texture, px, py)
|
|
Painter.setAlphaMatrix(texture, px, py, a);
|
|
if (a > before) {
|
|
a = (a - before) / (1 - before);
|
|
} else if (before) {
|
|
a = 0;
|
|
}
|
|
var result_color = Painter.combineColors(pxcolor, color, a);
|
|
if (Painter.lock_alpha) result_color.a = pxcolor.a
|
|
return result_color;
|
|
})
|
|
} else if (tool === 'eraser') {
|
|
Painter.editCircle(ctx, x, y, size, softness, function(pxcolor, opacity, px, py) {
|
|
if (Painter.lock_alpha) return pxcolor;
|
|
|
|
var a = b_opacity * opacity;
|
|
|
|
var before = Painter.getAlphaMatrix(texture, px, py)
|
|
Painter.setAlphaMatrix(texture, px, py, a);
|
|
|
|
if (a > before) {
|
|
a = (a - before) / (1 - before);
|
|
} else if (before) {
|
|
a = 0;
|
|
}
|
|
pxcolor.a = Math.clamp(pxcolor.a * (1-a), 0, 1);
|
|
return pxcolor;
|
|
|
|
})
|
|
}
|
|
ctx.restore();
|
|
},
|
|
useFilltool(texture, ctx, x, y, area) {
|
|
|
|
var color = tinycolor(ColorPanel.get()).toRgb();
|
|
let b_opacity = BarItems.slider_brush_opacity.get()/100;
|
|
let tool = Toolbox.selected.id;
|
|
|
|
let {rect, uvFactorX, uvFactorY, w, h} = area;
|
|
|
|
ctx.fillStyle = tinycolor(ColorPanel.get()).setAlpha(b_opacity).toRgbString();
|
|
|
|
var fill_mode = BarItems.fill_mode.get()
|
|
var cube = Painter.current.cube;
|
|
if (cube && fill_mode === 'cube') {
|
|
for (var face in cube.faces) {
|
|
var tag = cube.faces[face]
|
|
ctx.beginPath();
|
|
if (tag.getTexture() === texture) {
|
|
var face_rect = getRectangle(
|
|
Math.floor(tag.uv[0] * uvFactorX),
|
|
Math.floor(tag.uv[1] * uvFactorY),
|
|
Math.ceil(tag.uv[2] * uvFactorX),
|
|
Math.ceil(tag.uv[3] * uvFactorY)
|
|
)
|
|
ctx.rect(face_rect.ax, face_rect.ay, face_rect.x, face_rect.y)
|
|
ctx.fill()
|
|
}
|
|
}
|
|
|
|
} else if (fill_mode === 'face') {
|
|
ctx.fill()
|
|
} else {
|
|
|
|
var pxcol = [];
|
|
var map = {}
|
|
Painter.scanCanvas(ctx, x, y, 1, 1, (x, y, px) => {
|
|
px.forEach((val, i) => {
|
|
pxcol[i] = val
|
|
})
|
|
})
|
|
Painter.scanCanvas(ctx, rect[0], rect[1], w, h, (x, y, px) => {
|
|
if (pxcol.equals(px)) {
|
|
if (!map[x]) map[x] = {}
|
|
map[x][y] = true
|
|
}
|
|
})
|
|
var scan_value = true;
|
|
if (fill_mode === 'color_connected') {
|
|
function checkPx(x, y) {
|
|
if (map[x] && map[x][y]) {
|
|
map[x][y] = false;
|
|
|
|
checkPx(x+1, y)
|
|
checkPx(x-1, y)
|
|
checkPx(x, y+1)
|
|
checkPx(x, y-1)
|
|
}
|
|
}
|
|
checkPx(x, y)
|
|
scan_value = false;
|
|
}
|
|
Painter.scanCanvas(ctx, rect[0], rect[1], w, h, (x, y, px) => {
|
|
if (map[x] && map[x][y] === scan_value) {
|
|
var pxcolor = {
|
|
r: px[0],
|
|
g: px[1],
|
|
b: px[2],
|
|
a: px[3]/255
|
|
}
|
|
var result_color = Painter.combineColors(pxcolor, color, b_opacity);
|
|
px[0] = result_color.r
|
|
px[1] = result_color.g
|
|
px[2] = result_color.b
|
|
if (!Painter.lock_alpha) px[3] = result_color.a*255
|
|
}
|
|
})
|
|
}
|
|
},
|
|
runMirrorBrush(texture, x, y, event, uvTag) {
|
|
if (uvTag && Painter.current.cube) {
|
|
let mirror_cube = Painter.getMirrorCube(Painter.current.cube);
|
|
if (mirror_cube) {
|
|
|
|
let uvFactorX = 1 / Project.texture_width * texture.img.naturalWidth;
|
|
let uvFactorY = 1 / Project.texture_height * texture.img.naturalHeight;
|
|
|
|
let face = Painter.current.face;
|
|
let side_face = (face === 'west' || face === 'east')
|
|
if (side_face) face = Face.opposite[face];
|
|
face = mirror_cube.faces[face];
|
|
|
|
if (side_face &&
|
|
uvTag[1] === face.uv[1] && uvTag[3] === face.uv[3] &&
|
|
Math.min(uvTag[0], uvTag[2]) === Math.min(face.uv[0], face.uv[2])
|
|
//same face
|
|
) return;
|
|
|
|
//calculate original point
|
|
var point_on_uv = [
|
|
x - Math.min(uvTag[0], uvTag[2]) * uvFactorX,
|
|
y - Math.min(uvTag[1], uvTag[3]) * uvFactorY,
|
|
]
|
|
//calculate new point
|
|
if (face.uv[0] > face.uv[0+2] == uvTag[0] > uvTag[0+2]) {
|
|
point_on_uv[0] = Math.max(face.uv[0], face.uv[0+2]) * uvFactorX - point_on_uv[0] - 1;
|
|
if (BarItems.slider_brush_size.get()%2 == 0 && Toolbox.selected.brushTool) point_on_uv[0] += 1
|
|
} else {
|
|
point_on_uv[0] = Math.min(face.uv[0], face.uv[0+2]) * uvFactorX + point_on_uv[0];
|
|
}
|
|
if (face.uv[1] > face.uv[1+2] == uvTag[1] > uvTag[1+2]) {
|
|
point_on_uv[1] = Math.min(face.uv[1], face.uv[1+2]) * uvFactorY + point_on_uv[1];
|
|
} else {
|
|
point_on_uv[1] = Math.max(face.uv[1], face.uv[1+2]) * uvFactorY - point_on_uv[1] - 1;
|
|
}
|
|
|
|
let cube = Painter.current.cube;
|
|
Painter.current.cube = mirror_cube;
|
|
Painter.useBrushlike(texture, ...point_on_uv, event, face.uv, true, true);
|
|
Painter.current.cube = cube;
|
|
}
|
|
}
|
|
},
|
|
drawBrushLine(texture, end_x, end_y, event, new_face, uv) {
|
|
var start_x = (Painter.current.x == undefined ? end_x : Painter.current.x);
|
|
var start_y = (Painter.current.y == undefined ? end_y : Painter.current.y);
|
|
|
|
var diff_x = end_x - start_x;
|
|
var diff_y = end_y - start_y;
|
|
|
|
var length = Math.sqrt(diff_x*diff_x + diff_y*diff_y)
|
|
if (new_face && !length) {
|
|
length = 1
|
|
}
|
|
var i = 1;
|
|
var x, y;
|
|
while (i <= length) {
|
|
x = Math.round(start_x + diff_x / length * i)
|
|
y = Math.round(start_y + diff_y / length * i)
|
|
Painter.useBrushlike(texture, x, y, event, uv, i < length-1);
|
|
i++;
|
|
}
|
|
},
|
|
useShapeTool(texture, x, y, event, uvTag) {
|
|
Painter.brushChanges = true;
|
|
|
|
texture.edit(function(canvas) {
|
|
var ctx = canvas.getContext('2d')
|
|
ctx.clearRect(0, 0, canvas.width, canvas.height);
|
|
ctx.drawImage(Painter.current.clear, 0, 0)
|
|
|
|
let color = tinycolor(ColorPanel.get()).toRgb();
|
|
let b_opacity = BarItems.slider_brush_opacity.get()/100;
|
|
var width = BarItems.slider_brush_size.get();
|
|
let shape = BarItems.draw_shape_type.get();
|
|
let hollow = shape.substr(-1) == 'h';
|
|
shape = shape.replace(/_h$/, '');
|
|
|
|
if (uvTag) {
|
|
var rect = Painter.editing_area = [
|
|
uvTag[0] / Project.texture_width * texture.img.naturalWidth,
|
|
uvTag[1] / Project.texture_height * texture.img.naturalHeight,
|
|
uvTag[2] / Project.texture_width * texture.img.naturalWidth,
|
|
uvTag[3] / Project.texture_height * texture.img.naturalHeight
|
|
]
|
|
} else {
|
|
var rect = Painter.editing_area = [0, 0, texture.img.naturalWidth, texture.img.naturalHeight]
|
|
}
|
|
for (var t = 0; t < 2; t++) {
|
|
if (rect[t] > rect[t+2]) {
|
|
[rect[t], rect[t+2]] = [rect[t+2], rect[t]]
|
|
}
|
|
rect[t] = Math.round(rect[t])
|
|
rect[t+2] = Math.round(rect[t+2])
|
|
}
|
|
var [w, h] = [rect[2] - rect[0], rect[3] - rect[1]]
|
|
|
|
|
|
|
|
let diff_x = x - Painter.startPixel[0];
|
|
let diff_y = y - Painter.startPixel[1];
|
|
|
|
if (event.shiftKey) {
|
|
let clamp = Math.floor((Math.abs(diff_x) + Math.abs(diff_y))/2);
|
|
diff_x = diff_x>0 ? clamp : -clamp;
|
|
diff_y = diff_y>0 ? clamp : -clamp;
|
|
}
|
|
|
|
if (shape === 'rectangle') {
|
|
ctx.strokeStyle = ctx.fillStyle = tinycolor(ColorPanel.get()).setAlpha(b_opacity).toRgbString();
|
|
ctx.lineWidth = width;
|
|
ctx.beginPath();
|
|
var rect = getRectangle(Painter.startPixel[0], Painter.startPixel[1], Painter.startPixel[0]+diff_x, Painter.startPixel[1]+diff_y);
|
|
|
|
if (hollow) {
|
|
ctx.rect(rect.ax+(width%2 ? 0.5 : 1), rect.ay+(width%2 ? 0.5 : 1), rect.x, rect.y);
|
|
ctx.stroke();
|
|
} else {
|
|
ctx.rect(rect.ax, rect.ay, rect.x+1, rect.y+1);
|
|
ctx.fill();
|
|
}
|
|
} else if (shape === 'ellipse') {
|
|
Painter.modifyCanvasSection(ctx, rect[0], rect[1], w, h, (changePixel) => {
|
|
//changePixel(0, 0, editPx)
|
|
function editPx(pxcolor) {
|
|
return Painter.combineColors(pxcolor, color, b_opacity);
|
|
}
|
|
if (hollow) {
|
|
let r_min = Math.trunc(-width/2);
|
|
let r_max = Math.ceil(width/2);
|
|
for (var diff_x_m = diff_x+r_min; diff_x_m < diff_x+r_max; diff_x_m++) {
|
|
for (var diff_y_m = diff_y+r_min; diff_y_m < diff_y+r_max; diff_y_m++) {
|
|
for (var i = 0; i < Math.abs(diff_x_m); i++) {
|
|
for (var j = 0; j < 4; j++) {
|
|
changePixel(
|
|
Painter.startPixel[0] + (j<2?1:-1) * i,
|
|
Painter.startPixel[1] + (j%2?1:-1) * Math.round(Math.cos(Math.asin(i / Math.abs(diff_x_m))) * diff_y_m),
|
|
editPx
|
|
)
|
|
}
|
|
}
|
|
for (var i = 0; i < Math.abs(diff_y_m); i++) {
|
|
for (var j = 0; j < 4; j++) {
|
|
changePixel(
|
|
Painter.startPixel[0] + (j<2?1:-1) * Math.round(Math.sin(Math.acos(i / Math.abs(diff_y_m))) * diff_x_m),
|
|
Painter.startPixel[1] + (j%2?1:-1) * i,
|
|
editPx
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
diff_x = Math.clamp(diff_x, -64, 64);
|
|
diff_y = Math.clamp(diff_y, -64, 64);
|
|
for (var i = 0; i <= Math.abs(diff_x); i++) {
|
|
let radius = Math.round(Math.cos(Math.asin(i / Math.abs(diff_x))) * Math.abs(diff_y))
|
|
for (var k = 0; k <= radius; k++) {
|
|
for (var j = 0; j < 4; j++) {
|
|
changePixel(
|
|
Painter.startPixel[0] + (j<2?1:-1) * i,
|
|
Painter.startPixel[1] + (j%2?1:-1) * k,
|
|
editPx
|
|
)
|
|
}
|
|
}
|
|
}
|
|
for (var i = 0; i <= Math.abs(diff_y); i++) {
|
|
let radius = Math.round(Math.sin(Math.acos(i / Math.abs(diff_y))) * Math.abs(diff_x))
|
|
for (var k = 0; k <= radius; k++) {
|
|
for (var j = 0; j < 4; j++) {
|
|
changePixel(
|
|
Painter.startPixel[0] + (j<2?1:-1) * k,
|
|
Painter.startPixel[1] + (j%2?1:-1) * i,
|
|
editPx
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
//Painter.editing_area = undefined;
|
|
|
|
}, {no_undo: true, use_cache: true});
|
|
},
|
|
colorPicker(texture, x, y) {
|
|
var ctx = Painter.getCanvas(texture).getContext('2d')
|
|
Painter.scanCanvas(ctx, x, y, 1, 1, (x, y, px) => {
|
|
var t = tinycolor({
|
|
r: px[0],
|
|
g: px[1],
|
|
b: px[2],
|
|
a: px[3]/256
|
|
})
|
|
ColorPanel.set(t)
|
|
})
|
|
},
|
|
// Util
|
|
combineColors(base, added, opacity) {
|
|
if (Math.isNumber(base)) base = Jimp.intToRGBA(base)
|
|
if (Math.isNumber(added)) added = Jimp.intToRGBA(added)
|
|
|
|
if (added.a*opacity == 1) return added
|
|
|
|
var original_a = added.a
|
|
added.a = added.a*opacity
|
|
|
|
var mix = {};
|
|
mix.a = limitNumber(1 - (1 - added.a) * (1 - base.a), 0, 1); // alpha
|
|
mix.r = Math.round((added.r * added.a / mix.a) + (base.r * base.a * (1 - added.a) / mix.a)); // red
|
|
mix.g = Math.round((added.g * added.a / mix.a) + (base.g * base.a * (1 - added.a) / mix.a)); // green
|
|
mix.b = Math.round((added.b * added.a / mix.a) + (base.b * base.a * (1 - added.a) / mix.a)); // blue
|
|
|
|
added.a = original_a
|
|
return mix;
|
|
},
|
|
getMirrorCube(cube) {
|
|
let center = Format.centered_grid ? 0 : 8;
|
|
let e = 0.002
|
|
if (cube.from[0]-center === center-cube.to[0] && !cube.rotation[1] && !cube.rotation[2]) {
|
|
return cube;
|
|
} else {
|
|
for (var cube2 of Cube.all) {
|
|
if (
|
|
cube.inflate === cube2.inflate &&
|
|
Math.epsilon(cube.from[2], cube2.from[2], e) && Math.epsilon(cube.to[2], cube2.to[2], e) &&
|
|
Math.epsilon(cube.from[1], cube2.from[1], e) && Math.epsilon(cube.to[1], cube2.to[1], e) &&
|
|
Math.epsilon(cube.size(0), cube2.size(0), e) && Math.epsilon(cube.to[0]-center, center-cube2.from[0], e)
|
|
) {
|
|
return cube2;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
updateNslideValues() {
|
|
BarItems.slider_brush_size.update()
|
|
BarItems.slider_brush_softness.update()
|
|
BarItems.slider_brush_opacity.update()
|
|
},
|
|
getCanvas(texture) {
|
|
var c = document.createElement('canvas')
|
|
var ctx = c.getContext('2d');
|
|
c.width = texture.width;
|
|
c.height = texture.height;
|
|
ctx.drawImage(texture instanceof Texture ? texture.img : texture, 0, 0)
|
|
return c;
|
|
},
|
|
scanCanvas(ctx, x, y, w, h, cb) {
|
|
var arr = ctx.getImageData(x, y, w, h)
|
|
for (var i = 0; i < arr.data.length; i += 4) {
|
|
var pixel = arr.data.slice(i, i+4)
|
|
|
|
var px = x + (i/4) % w
|
|
var py = y + Math.floor((i/4) / w)
|
|
if (px >= ctx.canvas.width || px < 0 || py >= ctx.canvas.height || py < 0) continue;
|
|
pixel = cb(px, py, pixel)||pixel
|
|
|
|
pixel.forEach((p, pi) => {
|
|
arr.data[i+pi] = p
|
|
})
|
|
}
|
|
ctx.putImageData(arr, x, y)
|
|
},
|
|
modifyCanvasSection(ctx, x, y, w, h, cb) {
|
|
var arr = ctx.getImageData(x, y, w, h)
|
|
var processed = [];
|
|
|
|
cb((px, py, editPx) => {
|
|
//changePixel
|
|
px = Math.floor(px)-x;
|
|
py = Math.floor(py)-y;
|
|
if (px < 0 || px >= w) return;
|
|
if (py < 0 || py >= h) return;
|
|
let start = (px + py*w) * 4;
|
|
if (processed.includes(start)) return;
|
|
processed.push(start);
|
|
var result_color = editPx({
|
|
r: arr.data[start+0],
|
|
g: arr.data[start+1],
|
|
b: arr.data[start+2],
|
|
a: arr.data[start+3]/255
|
|
})
|
|
arr.data[start+0] = result_color.r
|
|
arr.data[start+1] = result_color.g
|
|
arr.data[start+2] = result_color.b
|
|
arr.data[start+3] = result_color.a*255
|
|
})
|
|
|
|
ctx.putImageData(arr, x, y)
|
|
},
|
|
drawRectangle(image, color, rect) {
|
|
var color = Jimp.intToRGBA(color)
|
|
image.scan(rect.x, rect.y, rect.w, rect.h, function (x, y, idx) {
|
|
this.bitmap.data[idx + 0] = color.r
|
|
this.bitmap.data[idx + 1] = color.g
|
|
this.bitmap.data[idx + 2] = color.b
|
|
this.bitmap.data[idx + 3] = color.a
|
|
});
|
|
},
|
|
editCircle(ctx, x, y, r, s, editPx) {
|
|
r = Math.round(r+1)/2
|
|
Painter.scanCanvas(ctx, x-Math.ceil(r)-2, y-Math.ceil(r)-2, 2*r+3, 2*r+3, function (px, py, pixel) {
|
|
if (
|
|
settings.paint_side_restrict.value &&
|
|
Painter.editing_area &&
|
|
typeof Painter.editing_area === 'object' &&
|
|
(
|
|
px+0.02 < Math.floor(Painter.editing_area[0]) ||
|
|
py+0.02 < Math.floor(Painter.editing_area[1]) ||
|
|
px+0.02 >= Painter.editing_area[2] ||
|
|
py+0.02 >= Painter.editing_area[3]
|
|
)
|
|
) {
|
|
return;
|
|
}
|
|
|
|
px -= x - r%1;
|
|
py -= y - r%1;
|
|
|
|
var distance = Math.sqrt(px*px + py*py)
|
|
if (s*r != 0) {
|
|
var pos_on_gradient = (distance-(1-s)*r) / (s*r)
|
|
} else {
|
|
var pos_on_gradient = Math.floor((distance*1.2)/r)
|
|
}
|
|
|
|
var opacity = limitNumber(1-pos_on_gradient, 0, 1)
|
|
|
|
if (opacity > 0) {
|
|
var result_color = editPx({
|
|
r: pixel[0],
|
|
g: pixel[1],
|
|
b: pixel[2],
|
|
a: pixel[3]/255
|
|
}, opacity, px+x, py+y)
|
|
pixel[0] = result_color.r
|
|
pixel[1] = result_color.g
|
|
pixel[2] = result_color.b
|
|
pixel[3] = result_color.a*255
|
|
}
|
|
});
|
|
},
|
|
drawRotatedRectangle(image, color, rect, cx, cy, angle) {
|
|
var color = Jimp.intToRGBA(color)
|
|
var sin = Math.sin(-Math.degToRad(angle))
|
|
var cos = Math.cos(-Math.degToRad(angle))
|
|
function rotatePoint(px, py) {
|
|
px -= cx
|
|
py -= cy
|
|
return {
|
|
x: (px * cos - py * sin) + cx,
|
|
y: (px * sin + py * cos) + cy
|
|
}
|
|
}
|
|
image.scan(0, 0, 48, 48, function (px, py, idx) {
|
|
var rotated = rotatePoint(px, py)
|
|
if (
|
|
rotated.x > rect.x-1 && rotated.x < rect.x + rect.w+2 &&
|
|
rotated.y > rect.y-1 && rotated.y < rect.y + rect.h+2
|
|
) {
|
|
var opacity = limitNumber(rect.x - rotated.x, 0, 1) +
|
|
limitNumber(rotated.x - (rect.x + rect.w), 0, 1) +
|
|
limitNumber(rect.y - rotated.y, 0, 1) +
|
|
limitNumber(rotated.y - (rect.y + rect.h), 0, 1)
|
|
|
|
opacity = 1-limitNumber(opacity*1.61, 0, 1)
|
|
if (this.bitmap.data[idx + 3]) {
|
|
opacity = 1
|
|
}
|
|
|
|
this.bitmap.data[idx + 0] = color.r
|
|
this.bitmap.data[idx + 1] = color.g
|
|
this.bitmap.data[idx + 2] = color.b
|
|
this.bitmap.data[idx + 3] = color.a*opacity
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
|
|
BARS.defineActions(function() {
|
|
|
|
new Tool('brush_tool', {
|
|
icon: 'fa-paint-brush',
|
|
category: 'tools',
|
|
toolbar: 'brush',
|
|
alt_tool: 'color_picker',
|
|
cursor: 'crosshair',
|
|
selectFace: true,
|
|
transformerMode: 'hidden',
|
|
paintTool: true,
|
|
brushTool: true,
|
|
allowWireframe: false,
|
|
keybind: new Keybind({key: 66}),
|
|
modes: ['paint'],
|
|
onCanvasClick: function(data) {
|
|
Painter.startPaintToolCanvas(data, data.event)
|
|
},
|
|
onSelect: function() {
|
|
Painter.updateNslideValues()
|
|
},
|
|
onUnselect: function() {
|
|
uv_dialog.all_editors.forEach(editor => {
|
|
editor.brush_outline.detach()
|
|
})
|
|
}
|
|
})
|
|
new Tool('fill_tool', {
|
|
icon: 'format_color_fill',
|
|
category: 'tools',
|
|
toolbar: 'brush',
|
|
alt_tool: 'color_picker',
|
|
cursor: 'crosshair',
|
|
selectFace: true,
|
|
transformerMode: 'hidden',
|
|
paintTool: true,
|
|
allowWireframe: false,
|
|
modes: ['paint'],
|
|
onCanvasClick: function(data) {
|
|
Painter.startPaintToolCanvas(data, data.event)
|
|
},
|
|
onSelect: function() {
|
|
Painter.updateNslideValues()
|
|
}
|
|
})
|
|
new Tool('eraser', {
|
|
icon: 'fa-eraser',
|
|
category: 'tools',
|
|
toolbar: 'brush',
|
|
selectFace: true,
|
|
transformerMode: 'hidden',
|
|
cursor: 'crosshair',
|
|
paintTool: true,
|
|
brushTool: true,
|
|
allowWireframe: false,
|
|
modes: ['paint'],
|
|
keybind: new Keybind({key: 69}),
|
|
onCanvasClick: function(data) {
|
|
Painter.startPaintToolCanvas(data, data.event)
|
|
},
|
|
onSelect: function() {
|
|
Painter.updateNslideValues()
|
|
}
|
|
})
|
|
new Tool('color_picker', {
|
|
icon: 'colorize',
|
|
category: 'tools',
|
|
toolbar: 'brush',
|
|
cursor: 'crosshair',
|
|
selectFace: true,
|
|
transformerMode: 'hidden',
|
|
paintTool: true,
|
|
allowWireframe: false,
|
|
modes: ['paint'],
|
|
onCanvasClick: function(data) {
|
|
Painter.startPaintToolCanvas(data, data.event)
|
|
},
|
|
onSelect: function() {
|
|
Painter.updateNslideValues()
|
|
}
|
|
})
|
|
new Tool('draw_shape_tool', {
|
|
icon: 'fas.fa-shapes',
|
|
category: 'tools',
|
|
toolbar: 'brush',
|
|
alt_tool: 'color_picker',
|
|
cursor: 'crosshair',
|
|
selectFace: true,
|
|
transformerMode: 'hidden',
|
|
paintTool: true,
|
|
allowWireframe: false,
|
|
modes: ['paint'],
|
|
condition: {modes: ['paint']},
|
|
keybind: new Keybind({key: 85}),
|
|
onCanvasClick: function(data) {
|
|
Painter.startPaintToolCanvas(data, data.event)
|
|
},
|
|
onSelect: function() {
|
|
Painter.updateNslideValues()
|
|
}
|
|
})
|
|
new Tool('copy_paste_tool', {
|
|
icon: 'fa-vector-square',
|
|
category: 'tools',
|
|
toolbar: 'brush',
|
|
alt_tool: 'color_picker',
|
|
cursor: 'crosshair',
|
|
selectFace: true,
|
|
transformerMode: 'hidden',
|
|
paintTool: true,
|
|
allowWireframe: false,
|
|
modes: ['paint'],
|
|
condition: {modes: ['paint']},
|
|
keybind: new Keybind({key: 77})
|
|
})
|
|
|
|
new BarSelect('draw_shape_type', {
|
|
category: 'paint',
|
|
condition: () => Toolbox && Toolbox.selected.id === 'draw_shape_tool',
|
|
onChange() {
|
|
BARS.updateConditions();
|
|
Painter.updateNslideValues()
|
|
},
|
|
options: {
|
|
rectangle: true,
|
|
rectangle_h: true,
|
|
ellipse: true,
|
|
ellipse_h: true,
|
|
}
|
|
})
|
|
new BarSelect('fill_mode', {
|
|
category: 'paint',
|
|
condition: () => Toolbox && Toolbox.selected.id === 'fill_tool',
|
|
options: {
|
|
face: true,
|
|
cube: true,
|
|
color_connected: true,
|
|
color: true,
|
|
}
|
|
})
|
|
new Action('mirror_painting', {
|
|
label: true,
|
|
icon: 'check_box_outline_blank',
|
|
category: 'paint',
|
|
condition: () => Modes.paint,
|
|
click: function () {
|
|
Painter.mirror_painting = !Painter.mirror_painting;
|
|
this.setIcon(Painter.mirror_painting ? 'check_box' : 'check_box_outline_blank')
|
|
}
|
|
})
|
|
new Action('lock_alpha', {
|
|
icon: 'fas.fa-unlock',
|
|
category: 'paint',
|
|
condition: () => Modes.paint,
|
|
click: function () {
|
|
Painter.lock_alpha = !Painter.lock_alpha;
|
|
this.setIcon(Painter.lock_alpha ? 'fas.fa-lock' : 'fas.fa-unlock')
|
|
}
|
|
})
|
|
|
|
new Action('painting_grid', {
|
|
name: tl('settings.painting_grid'),
|
|
description: tl('settings.painting_grid.desc'),
|
|
icon: 'check_box',
|
|
icon_states: ['grid_off', 'grid_on'],
|
|
category: 'view',
|
|
condition: () => Modes.paint,
|
|
keybind: new Keybind({key: 71}),
|
|
linked_setting: 'painting_grid',
|
|
click: function () {
|
|
BarItems.painting_grid.toggleLinkedSetting()
|
|
}
|
|
})
|
|
|
|
new NumSlider('slider_brush_size', {
|
|
condition: () => (Toolbox && ['brush_tool', 'eraser', 'draw_shape_tool'].includes(Toolbox.selected.id)),
|
|
tool_setting: 'brush_size',
|
|
category: 'paint',
|
|
settings: {
|
|
min: 1, max: 50, interval: 1, default: 1,
|
|
}
|
|
})
|
|
new NumSlider('slider_brush_softness', {
|
|
category: 'paint',
|
|
condition: () => (Toolbox && ['brush_tool', 'eraser'].includes(Toolbox.selected.id)),
|
|
tool_setting: 'brush_softness',
|
|
settings: {
|
|
min: 0, max: 100, default: 0,
|
|
interval: function(event) {
|
|
if (event.shiftKey && event.ctrlOrCmd) {
|
|
return 0.25;
|
|
} else if (event.shiftKey) {
|
|
return 5;
|
|
} else if (event.ctrlOrCmd) {
|
|
return 1;
|
|
} else {
|
|
return 10;
|
|
}
|
|
}
|
|
}
|
|
})
|
|
new NumSlider('slider_brush_opacity', {
|
|
category: 'paint',
|
|
condition: () => (Toolbox && ['brush_tool', 'eraser', 'fill_tool', 'draw_shape_tool'].includes(Toolbox.selected.id)),
|
|
tool_setting: 'brush_opacity',
|
|
settings: {
|
|
min: 0, max: 100, default: 100,
|
|
interval: function(event) {
|
|
if (event.shiftKey && event.ctrlOrCmd) {
|
|
return 0.25;
|
|
} else if (event.shiftKey) {
|
|
return 5;
|
|
} else if (event.ctrlOrCmd) {
|
|
return 1;
|
|
} else {
|
|
return 10;
|
|
}
|
|
}
|
|
}
|
|
})
|
|
})
|