Как сделать так, чтобы векторы отображались между точками на сетке? - PullRequest
1 голос
/ 04 апреля 2019

Я в настоящее время борюсь с моим переходом на HTML5 и Java, после того, как в основном только Python и C #.

Я нахожусь в процессе (пытаюсь) создать точечную сетку, см. Код ниже, который рисует линии (см. Код ниже), между точками при нажатии, с конечным результатом, который я мог отобразитьплан этажа, используя его (я строитель по профессии).

Я также пытаюсь решить, как удалять нарисованные линии по отдельности, не нажимая кнопку «сброс», может быть, щелкнув по ним правой кнопкой мыши.

В любом случае, из-за того, что такой новыйэто вызывает у меня сильную головную боль.Я возился с paper.js и jquery, и уже несколько дней мне не везет.

Может ли кто-нибудь из вас указать мне правильное направление для учебника / примера того, что мне нужно, или какого-либо совета о том, где искать помощь?

В противном случае, как можно сделать«точка» на этой сетке «кликабельна» и соответствует ли она начальной и конечной точке вектора?

Я включил код, который я работаю снизу.

Точечная сетка:

https://codepen.io/veljamatic/pen/pypxRR

Вектор, с использованием paper.js (из приведенного здесь учебника, конечный результат внизу страницы http://paperjs.org/tutorials/geometry/vector-geometry/):

var values = {
    fixLength: false,
    fixAngle: false,
    showCircle: false,
    showAngleLength: true,
    showCoordinates: false
};

var vectorStart, vector, vectorPrevious;
var vectorItem, items, dashedItems;

function processVector(event, drag) {
    vector = event.point - vectorStart;
    if (vectorPrevious) {
        if (values.fixLength && values.fixAngle) {
        vector = vectorPrevious;
        } else if (values.fixLength) {
        vector.length = vectorPrevious.length;
        } else if (values.fixAngle) {
            vector = vector.project(vectorPrevious);
        }
    }
    drawVector(drag);
}

function drawVector(drag) {
    if (items) {
        for (var i = 0, l = items.length; i < l; i++) {
            items[i].remove();
        }
    }
    if (vectorItem)
        vectorItem.remove();
    items = [];
    var arrowVector = vector.normalize(10);
    var end = vectorStart + vector;
    vectorItem = new Group([
        new Path([vectorStart, end]),
        new Path([
            end + arrowVector.rotate(135),
            end,
            end + arrowVector.rotate(-135)
        ])
    ]);
    vectorItem.strokeWidth = 0.75;
    vectorItem.strokeColor = '#e4141b';
    // Display:
    dashedItems = [];
    // Draw Circle
    if (values.showCircle) {
        dashedItems.push(new Path.Circle({
            center: vectorStart,
            radius: vector.length
        }));
    }
    // Draw Labels
    if (values.showAngleLength) {
        drawAngle(vectorStart, vector, !drag);
        if (!drag)
            drawLength(vectorStart, end,                                vector.angle < 0 ? -1 : 1, true);
}
var quadrant = vector.quadrant;
if (values.showCoordinates && !drag) {
    drawLength(vectorStart, vectorStart + [vector.x, 0],
            [1, 3].indexOf(quadrant) != -1 ? -1 : 1, true, vector.x, 'x: ');
    drawLength(vectorStart, vectorStart + [0, vector.y], 
            [1, 3].indexOf(quadrant) != -1 ? 1 : -1, true, vector.y, 'y: ');
}
for (var i = 0, l = dashedItems.length; i < l; i++) {
    var item = dashedItems[i];
    item.strokeColor = 'black';
    item.dashArray = [1, 2];
    items.push(item);
}
// Update palette
values.x = vector.x;
values.y = vector.y;
values.length = vector.length;
values.angle = vector.angle;
}

function drawAngle(center, vector, label) {
    var radius = 25, threshold = 10;
    if (vector.length < radius + threshold ||   Math.abs(vector.angle) < 15)
        return;
    var from = new Point(radius, 0);
    var through = from.rotate(vector.angle / 2);
    var to = from.rotate(vector.angle);
    var end = center + to;
    dashedItems.push(new Path.Line(center,
            center + new Point(radius + threshold,  0)));
    dashedItems.push(new Path.Arc(center + from, center +   through, end));
    var arrowVector = to.normalize(7.5).rotate(vector.angle < 0 ? -90 : 90);
    dashedItems.push(new Path([
            end + arrowVector.rotate(135),
            end,
            end + arrowVector.rotate(-135)
    ]));
    if (label) {
        // Angle Label
        var text = new PointText(center
                + through.normalize(radius + 10)    + new Point(0, 3));
        text.content = Math.floor(vector.angle * 100) /     100 + '°';
        text.fillColor = 'black';
        items.push(text);
    }
}

function drawLength(from, to, sign, label, value, prefix) {
    var lengthSize = 5;
    if ((to - from).length < lengthSize * 4)
        return;
    var vector = to - from;
    var awayVector = vector.normalize(lengthSize).rotate(90     * sign);
    var upVector = vector.normalize(lengthSize).rotate(45 *     sign);
    var downVector = upVector.rotate(-90 * sign);
    var lengthVector = vector.normalize(
            vector.length / 2 - lengthSize *    Math.sqrt(2));
    var line = new Path();
    line.add(from + awayVector);
    line.lineBy(upVector);
    line.lineBy(lengthVector);
    line.lineBy(upVector);
    var middle = line.lastSegment.point;
    line.lineBy(downVector);
    line.lineBy(lengthVector);
    line.lineBy(downVector);
    dashedItems.push(line);
    if (label) {
        // Length Label
        var textAngle = Math.abs(vector.angle) > 90
                ? textAngle = 180 + vector.angle    : vector.angle;
        // Label needs to move away by different amounts    based on the
        // vector's quadrant:
        var away = (sign >= 0 ? [1, 4] : [2,    3]).indexOf(vector.quadrant) != -1
                ? 8 : 0;
        value = value || vector.length;
        var text = new PointText({
            point: middle +     awayVector.normalize(away + lengthSize),
            content: (prefix || '') +   Math.floor(value * 1000) / 1000,
            fillColor: 'black',
            justification: 'center'
        });
        text.rotate(textAngle);
        items.push(text);
    }
}

var dashItem;

function onMouseDown(event) {
    var end = vectorStart + vector;
    var create = false;
    if (event.modifiers.shift && vectorItem) {
        vectorStart = end;
        create = true;
    } else if (vector && (event.modifiers.option
            || end && end.getDistance(event.point) <    10)) {
        create = false;
    } else {
        vectorStart = event.point;
    }
    if (create) {
        dashItem = vectorItem;
        vectorItem = null;
    }
    processVector(event, true);
//  document.redraw();
}

function onMouseDrag(event) {
    if (!event.modifiers.shift && values.fixLength &&   values.fixAngle)
        vectorStart = event.point;
    processVector(event, event.modifiers.shift);
}

function onMouseUp(event) {
    processVector(event, false);
    if (dashItem) {
        dashItem.dashArray = [1, 2];
        dashItem = null;
    }
    vectorPrevious = vector;
}

1 Ответ

1 голос
/ 04 апреля 2019

Пожалуйста, прочитайте комментарии в моем коде.

К этому вы можете добавить меню кнопок: Чтобы нарисовать последнюю линию, вам нужно удалить последнюю точку в массиве точек. Чтобы начать новый набор линий, вы можете создать массивмассивы точек.

Надеюсь, это поможет вам начать ваш проект.

const ctx = canvas.getContext("2d");
let cw = (canvas.width = window.innerWidth);
let ch = (canvas.height = window.innerWidth);
//the size of the cell relative to the width of the window
let size = cw / 50;
// the cells array
let cells = [];
// the points array.
let points = [];

class Cell {
  constructor(x, y, r) {
    this.x = x;
    this.y = y;
    this.r = r;
  }

  draw() {
    ctx.beginPath();
    ctx.moveTo(this.x - this.r, this.y - this.r);
    ctx.lineTo(this.x + this.r, this.y - this.r);
    ctx.lineTo(this.x + this.r, this.y + this.r);
    ctx.lineTo(this.x - this.r, this.y + this.r);
    ctx.closePath();
    //no stroke 
  }
  center() {
    // this is drawing a dot in the middle of the cell
    ctx.beginPath();
    ctx.arc(this.x, this.y, 2, 0, 2 * Math.PI);
    ctx.fill();
  }
}


// draw the grid of cells
for (let y = size / 2; y < ch; y += 2 * size) {
  for (let x = size / 2; x < cw; x += 2 * size) {
    cells.push(new Cell(x, y, size));
  }
}

cells.forEach(c => {
  //uncomment to see the cells
  //c.draw()
  //ctx.stroke();
  c.center();
});



canvas.addEventListener("click", evt => {
  // the mouse position
  let m = oMousePos(canvas, evt);
  // clear the context
  ctx.clearRect(0, 0, cw, ch);
  //redraw every center
  cells.forEach(c => {
    c.center();
  });

  
  for (let i = 0; i < cells.length; i++) {
    cells[i].draw();
    //check if the mouse is inside the cell
    if (ctx.isPointInPath(m.x, m.y)) {
      // marc the clicked cel with a red dot
      ctx.fillStyle = "red";
      cells[i].center();
      //store the number of the clicked cell in the points array
      points.push(i); 
      ctx.fillStyle = "black";
      //exit the loop
      break;
    }
  }
if(points.length >0){drawLines();}
});



function drawLines() {
  ctx.beginPath();
  ctx.strokeStyle = "black";
  ctx.moveTo(cells[points[0]].x, cells[points[0]].y);
  for (let i = 1; i < points.length; i++) {
    ctx.lineTo(cells[points[i]].x, cells[points[i]].y);
  }
  ctx.stroke();
}


// a function to detect the mouse position
function oMousePos(canvas, evt) {
  var ClientRect = canvas.getBoundingClientRect();
  return {
    x: Math.round(evt.clientX - ClientRect.left),
    y: Math.round(evt.clientY - ClientRect.top)
  };
}

function Init() {
  // get again the size of the canvas
  cw = canvas.width = window.innerWidth;
  ch = canvas.height = window.innerWidth;
  //the size of every cell
  size = cw / 50;
  // empty the cells array
  cells = [];
  // draw again the grid of cells
  for (let y = size / 2; y < ch; y += 2 * size) {
    for (let x = size / 2; x < cw; x += 2 * size) {
      cells.push(new Cell(x, y, size));
    }
  }
  //draw the center of every cell
  cells.forEach(c => {
    c.center();
  });
  //draw the lines
  if(points.length >0){drawLines();}
}



window.setTimeout(function() {
  Init();
  window.addEventListener("resize", Init, false);
}, 15);
canvas{background:#efefef}
<canvas id="canvas"></canvas>
...