FileAttachmentDynamic = function(filename) {
return new Function("FileAttachment", `return FileAttachment("${filename}")`)(FileAttachment)
}
// Vaihdetaan tekstin väri
color = function(vari) {
if(vari > 0){
return "green"
}else{
return "red"
}
}
// Pyöristys
pyoristys = function(n, places) {
if (!places) return Math.round(n);
const d = 10 ** places;
return Math.round(n * d) / d;
}
//Euromerkintä
eurot = function(number) {
return number.toLocaleString('fi-FI', { style: 'currency', currency: 'EUR',minimumFractionDigits: 0 });
}
// Euromuutoksen laskeminen kertoimella
muutoskerroin = function(x, y) {
return (1+x/100)*y;
}
// Muutosprosentti
muutosprosentti = function(x, y) {
return pyoristys(((x-y)/y)*100,1);
}
// Tuhatjakaja
tuhatjakaja = function(x) {
return x/1000;
}
// Euromuutoksen laskeminen kertoimella
palkkakerroin = function(x, y) {
return 100*((1+x/100)*(1+y/100)-1);
}
// Jako-osuuden asukaskerroin
asukaskerroin = function(x, y) {
return (x/y);
}
// Jako-osuuden veroprosenttikerroin miinus1/miinus3 vuotta miinus 1 vuoden kerroin
veroprosenttikerroin = function(x, y) {
return (x/y);
}
// Kunnan muok. kunnallisvero, alkuvuosi ja loppuvuosi
kunnallisvero_muokkaus = function(kunnallisvero,asukaskerroin,veroprosentti) {
return kunnallisvero*asukaskerroin*veroprosentti;
}
//Tekstimuutokset
fn_nousu_lasku_txt = function(arvo){
if(arvo > 0) {
return "kasvanut"
}
else {
return "laskenut";
}
}
//TARVITAAN KARTOISSA
serialize = {
const xmlns = "http://www.w3.org/2000/xmlns/";
const xlinkns = "http://www.w3.org/1999/xlink";
const svgns = "http://www.w3.org/2000/svg";
return function serialize(svg) {
svg = svg.cloneNode(true);
const fragment = window.location.href + "#";
const walker = document.createTreeWalker(svg, NodeFilter.SHOW_ELEMENT);
while (walker.nextNode()) {
for (const attr of walker.currentNode.attributes) {
if (attr.value.includes(fragment)) {
attr.value = attr.value.replace(fragment, "#");
}
}
}
svg.setAttributeNS(xmlns, "xmlns", svgns);
svg.setAttributeNS(xmlns, "xmlns:xlink", xlinkns);
const serializer = new window.XMLSerializer;
const string = serializer.serializeToString(svg);
return new Blob([string], {type: "image/svg+xml"});
};
}
function toSVG(chart) {
if (chart.nodeName !== "FIGURE") {
return chart;
}
// the chart needs to be in the body if we want to read values, positions, sizes…
document.body.appendChild(chart);
const [x0, y0, width, height] = getBounds([chart]);
const nodes = [];
for (const node of d3
.select(chart)
.selectChildren("h1,h2,h3,div,figcaption,svg")) {
switch (node.nodeName.toLowerCase()) {
case "div":
{
const children = d3.select(node).selectChildren("div,span");
const height = getBounds([node, ...children])[3] + 2;
const svg = d3
.select(chart)
.append("svg")
.attr("width", width)
.attr("height", height);
nodes.push(svg.node());
const swatches = svg
.selectAll()
.data(
Array.from(children, (d) => {
const svg = d3.select(d).select("svg").node();
const bbox = svg.getBBox();
return {
style: window.getComputedStyle(d),
svg,
width: bbox.width,
height: bbox.height,
text: d.textContent,
bounds: getBounds([d])
};
})
)
.join("g")
.attr(
"transform",
(d) => `translate(${d.bounds[0] - x0},${10 + d.bounds[1] - y0})`
);
swatches
.append((d) => d.svg) // "rect")
.attr("width", (d) => d.width)
.attr("height", (d) => d.height)
.attr("y", (d) => `${-parseFloat(d.height) / 2}px`);
swatches
.append("text")
.text((d) => d.text)
.attr("x", (d) => d.width)
.attr("dx", 5)
.attr("dy", "0.38em")
.attr("font-family", (d) => d.style.fontFamily)
.attr("font-size", (d) => d.style.fontSize)
.attr("fill", (d) => d.style.color);
}
break;
case "figcaption":
case "h1":
case "h2":
case "h3":
{
const svg = d3
.select(chart)
.append("svg")
.attr("width", width)
.attr("overflow", "visible");
nodes.push(svg.node());
const children = d3.select(node).selectChildren();
let h = 0;
for (const d of children.size() > 0
? children.selectChildren()
: [node]) {
const style = window.getComputedStyle(d);
const t = svg
.append("g")
.attr("transform", `translate(0,${h})`)
.append(() =>
d3
.select(
Plot.text([d.textContent], {
text: (d) => d,
lineWidth:
(1.06 * parseFloat(style.width)) /
parseFloat(style.fontSize),
lineHeight: 1.2,
frameAnchor: "top-left"
}).plot()
)
.select("text")
.attr("font-family", style.fontFamily)
.attr("font-size", 1.08 * parseFloat(style.fontSize))
.attr("font-weight", style.fontWeight)
.attr("fill", style.color)
.node()
);
h += getBounds([t.node()])[3] + 4;
}
svg.attr("height", h);
}
break;
case "svg":
d3.select(chart).append(() => node);
nodes.push(node);
break;
}
}
return serializeAll(nodes)
.then((blob) => blob.text())
.then((c) => {
document.body.removeChild(chart);
return Object.assign(svg`${c}`, chart);
});
}
// Given an array of SVG elements, composites them into a single SVG element,
// and then serializes the result to a blob.
async function serializeAll(elements, {padding = 10} = {}) {
const fragment = location.href + "#";
let root;
if (elements.length === 1) {
root = elements[0].cloneNode(true); // optimize common case
} else {
const [ox, oy, dx, dy] = getBounds(elements);
root = document.createElementNS(svgns, "svg");
root.setAttribute("width", dx + 2 * padding);
root.setAttribute("height", dy + 2 * padding);
root.setAttribute("viewBox", [-padding, -padding, dx + 2 * padding, dy + 2 * padding]);
for (const element of elements) {
const svg = root.appendChild(element.cloneNode(true));
const { x, y, width, height } = element.getBoundingClientRect();
svg.setAttribute("x", x - ox);
svg.setAttribute("y", y - oy);
svg.setAttribute("width", width);
svg.setAttribute("height", height);
}
}
const walker = document.createTreeWalker(root, NodeFilter.SHOW_ELEMENT);
while (walker.nextNode()) {
const node = walker.currentNode;
for (const attr of node.attributes) {
if (attr.value.includes(fragment)) {
attr.value = attr.value.replace(fragment, "#");
}
}
}
root.setAttributeNS(xmlns, "xmlns", svgns);
root.setAttributeNS(xmlns, "xmlns:xlink", xlinkns);
const serializer = new XMLSerializer();
const string = serializer.serializeToString(root);
return new Blob([string], { type: "image/svg+xml" });
}
function getBounds(elements) {
let x1 = Infinity;
let y1 = x1;
let x2 = -x1;
let y2 = x2;
for (const element of elements) {
const { x, y, width, height } = element.getBoundingClientRect();
if (x < x1) x1 = x;
if (x + width > x2) x2 = x + width;
if (y < y1) y1 = y;
if (y + height > y2) y2 = y + height;
}
return [x1, y1, x2 - x1, y2 - y1];
}
// based on https://observablehq.com/@gka/cheap-fit-text-to-circle
function lines(text, targetWidth) {
const CHAR_W = {
"a":7,"B":8,"C":8,"c":6,"D":9,"f":4,"G":9,"H":9,"I":3,"i":3,"J":5,"j":3,"K":8,"k":6,
"l":3,"M":11,"m":11,"N":9,"O":9,"P":8,"Q":9,"R":8,"r":4,"S":8,"s":6,"t":4,"U":9,"v":6,
"W":11,"w":9,"x":6,"y":6,"z":5,".":2,",":2,":":2,";":2
};
function measureWidth(text) { return d3.sum(text, char => CHAR_W[char] || CHAR_W["a"]) * 0.8; };
const words = text.split(" ");
let line;
let lineWidth0 = Infinity;
const lines = [];
for (let i = 0, n = words.length; i < n; ++i) {
let lineText1 = (line ? line.text + " " : "") + words[i];
let lineWidth1 = measureWidth(lineText1);
if ((lineWidth0 + lineWidth1) / 2 < targetWidth) {
line.width = lineWidth0 = lineWidth1;
line.text = lineText1;
} else {
lineWidth0 = measureWidth(words[i]);
line = {width: lineWidth0, text: words[i]};
lines.push(line);
}
}
return lines;
}
xmlns = "http://www.w3.org/2000/xmlns/"
xlinkns = "http://www.w3.org/1999/xlink"
svgns = "http://www.w3.org/2000/svg"
Avainsanat
Kuntien kuukausiraportti, väestö, yhteistyö, avoin lähdekoodi, Tilastokeskus, tietojohtaminen
Aluetalous ja yritystoiminta
Yrityskanta
Yrityskanta
Yrityskannassa oli yrityksiä 2024Q2 yhteensä 678 (lukema 2024Q1: 686).
Vuosineljännes | Yrityskanta |
---|---|
2018Q3 | 639 |
2018Q4 | 645 |
2019Q1 | 642 |
2019Q2 | 646 |
2019Q3 | 653 |
2019Q4 | 656 |
2020Q1 | 661 |
2020Q2 | 661 |
2020Q3 | 660 |
2020Q4 | 663 |
2021Q1 | 669 |
2021Q2 | 671 |
2021Q3 | 677 |
2021Q4 | 678 |
2022Q1 | 682 |
2022Q2 | 685 |
2022Q3 | 683 |
2022Q4 | 685 |
2023Q1 | 689 |
2023Q2 | 683 |
2023Q3 | 680 |
2023Q4 | 686 |
2024Q1 | 686 |
2024Q2 | 678 |
Aloittaneet ja lopettaneet yritykset
Aloittaneet ja lopettaneet yritykset
Aloittaneita yrityksiä ajanjaksolla 2024Q2 oli yhteensä 5 (lukema 2024Q1: 7). . Lopettaneiden yritysten määrä oli samalla ajanjaksolla yhteensä 8 (lukema 2024Q1: 13)..
Vuosineljännes | Aloittaneet yritykset (lkm) | Lopettaneet yritykset (lkm) |
---|---|---|
2018Q3 | 9 | 5 |
2018Q4 | 11 | 10 |
2019Q1 | 9 | 6 |
2019Q2 | 10 | 4 |
2019Q3 | 11 | 6 |
2019Q4 | 9 | 5 |
2020Q1 | 14 | 9 |
2020Q2 | 9 | 7 |
2020Q3 | 6 | 4 |
2020Q4 | 7 | 6 |
2021Q1 | 10 | 7 |
2021Q2 | 9 | 3 |
2021Q3 | 9 | 5 |
2021Q4 | 6 | 5 |
2022Q1 | 9 | 8 |
2022Q2 | 11 | 8 |
2022Q3 | 6 | 6 |
2022Q4 | 8 | 5 |
2023Q1 | 8 | 14 |
2023Q2 | 8 | 10 |
2023Q3 | 7 | 2 |
2023Q4 | 8 | 7 |
2024Q1 | 7 | 13 |
2024Q2 | 5 | 8 |
Avoimet työpaikat kuukauden laskentapäivänä (lkm.)
Avoimet työpaikat kuukauden laskentapäivänä (lkm.)
Avoimia työpaikkoja oli 2024M09 yhteensä 35 (lukema 2024M08: 37).
Kuukausi | Avoimet työpaikat kuukauden laskentapäivänä (lkm.) |
---|---|
2022M10 | 113 |
2022M11 | 70 |
2022M12 | 52 |
2023M01 | 159 |
2023M02 | 190 |
2023M03 | 187 |
2023M04 | 108 |
2023M05 | 97 |
2023M06 | 34 |
2023M07 | 36 |
2023M08 | 31 |
2023M09 | 37 |
2023M10 | 63 |
2023M11 | 44 |
2023M12 | 70 |
2024M01 | 127 |
2024M02 | 113 |
2024M03 | 112 |
2024M04 | 74 |
2024M05 | 54 |
2024M06 | 35 |
2024M07 | 28 |
2024M08 | 37 |
2024M09 | 35 |
Uudet avoimet työpaikat
Avoimet työpaikat kuukauden laskentapäivänä (lkm.)
Avoimia työpaikkoja oli 2024M09 yhteensä 29 (lukema 2024M08: 27).
Kuukausi | Uudet avoimet työpaikat kuukauden aikana (lkm.) |
---|---|
2022M10 | 166 |
2022M11 | 134 |
2022M12 | 68 |
2023M01 | 186 |
2023M02 | 114 |
2023M03 | 97 |
2023M04 | 117 |
2023M05 | 104 |
2023M06 | 32 |
2023M07 | 30 |
2023M08 | 41 |
2023M09 | 54 |
2023M10 | 75 |
2023M11 | 28 |
2023M12 | 52 |
2024M01 | 85 |
2024M02 | 74 |
2024M03 | 61 |
2024M04 | 41 |
2024M05 | 54 |
2024M06 | 40 |
2024M07 | 6 |
2024M08 | 27 |
2024M09 | 29 |