{"id":2756,"date":"2022-04-25T15:44:22","date_gmt":"2022-04-25T13:44:22","guid":{"rendered":"https:\/\/www.nettipuhelinkauppa.fi\/?page_id=2756"},"modified":"2022-12-21T16:31:34","modified_gmt":"2022-12-21T14:31:34","slug":"gxp1630","status":"publish","type":"page","link":"https:\/\/www.perustele.fi\/index.php\/gxp1630\/","title":{"rendered":"GXP1630"},"content":{"rendered":"<div id=\"attachment_2450\" style=\"width: 160px\" class=\"wp-caption aligncenter\"><img loading=\"lazy\" decoding=\"async\" aria-describedby=\"caption-attachment-2450\" class=\"size-thumbnail wp-image-2450\" src=\"https:\/\/www.perustele.fi\/wp-content\/uploads\/2021\/12\/1630_front-150x150.png\" alt=\"\" width=\"150\" height=\"150\" srcset=\"https:\/\/www.perustele.fi\/wp-content\/uploads\/2021\/12\/1630_front-150x150.png 150w, https:\/\/www.perustele.fi\/wp-content\/uploads\/2021\/12\/1630_front-140x140.png 140w, https:\/\/www.perustele.fi\/wp-content\/uploads\/2021\/12\/1630_front-370x370.png 370w, https:\/\/www.perustele.fi\/wp-content\/uploads\/2021\/12\/1630_front-300x300.png 300w, https:\/\/www.perustele.fi\/wp-content\/uploads\/2021\/12\/1630_front-100x100.png 100w, https:\/\/www.perustele.fi\/wp-content\/uploads\/2021\/12\/1630_front-1200x1200-cropped.png 1200w\" sizes=\"auto, (max-width: 150px) 100vw, 150px\" \/><p id=\"caption-attachment-2450\" class=\"wp-caption-text\">GXP1630<\/p><\/div>\n<p>GXP1630 on varustettu joukolla helppok\u00e4ytt\u00f6isi\u00e4 VoIP-ominaisuuksia. GXP1630 on tarkoitettu ensisijaisesti pieniin ja keskisuuriin puhelum\u00e4\u00e4riin ja tehokkaaseen puhelujen k\u00e4sittelyyn. Siin\u00e4 on \u00a03 linjaa\/SIP-tili\u00e4 ja 8 kaksiv\u00e4rist\u00e4 BLF\/pikavalintan\u00e4pp\u00e4int\u00e4. GXP1630 tukee my\u00f6s parhaita mahdollisia yhteysnopeuksia ja puhelun laatua kahdella Gigabit-portilla ja HD-\u00e4\u00e4nell\u00e4 sek\u00e4 kaiutinpuhelimessa ett\u00e4 luurissa. Muiden ominaisuuksiensa, kuten integroidun PoE:n, 3 ohjelmoitavan XML-ohjelmiston\u00e4pp\u00e4imen ja 4-suuntaisen neuvottelutuen, ansiosta GXP1630 on korkealaatuinen ja monipuolinen perus IP -puhelin.<\/p>\n<p>&nbsp;<\/p>\n<p>Kuten kaikki Grandstream IP -puhelimet, GXP1630 sis\u00e4lt\u00e4\u00e4 huippuluokan suojaustekniikan (SRTP ja TLS). GXP1630 tukee useita automaattisia provisiointivaihtoehtoja, mukaan lukien zero config Grandstreamin UCM-sarjan IP-vaihteilla, salatut XML-tiedostot ja TR-069, jotta k\u00e4ytt\u00f6\u00f6notto olisi eritt\u00e4in helppoa.<\/p>\n<p style=\"text-align: center;\">Ominaisuuksia<\/p>\n<p style=\"text-align: center;\">3 SIP-tili\u00e4<\/p>\n<p style=\"text-align: center;\">3 linjaa<\/p>\n<p style=\"text-align: center;\">4-suuntainen neuvottelu<\/p>\n<p style=\"text-align: center;\">3 ohjelmoitavaa XML kontekstiherkk\u00e4\u00e4 ohjelman\u00e4pp\u00e4int\u00e4<\/p>\n<p style=\"text-align: center;\">HD-\u00e4\u00e4ni kaiuttimessa ja luurissa<\/p>\n<p style=\"text-align: center;\">Kaksikytketyt Gigabit-portit<\/p>\n<p style=\"text-align: center;\">Integroitu PoE<\/p>\n<p style=\"text-align: center;\">8 kaksiv\u00e4rist\u00e4 BLF\/lyhytvalintan\u00e4pp\u00e4int\u00e4<\/p>\n<p style=\"text-align: center;\">EHS-tuki Plantronics-kuulokkeille<\/p>\n<p style=\"text-align: center;\">Jopa 1000 yhteystietoa<\/p>\n<p style=\"text-align: center;\">Puheluhistoria jopa 200 tietuetta<\/p>\n<p>Lis\u00e4tietopyynt\u00f6:<\/p>\n<form class=\"accua-form __accua-form__1 accua-form-view-sidebyside\" novalidate=\"novalidate\" action=\"#\" id=\"accua-form___accua-form__1_69ea045000f29\" method=\"post\"><input type=\"hidden\" name=\"_AccuaForm_ID\" value=\"__accua-form__1\" id=\"accua-form___accua-form__1_69ea045000f29-element-0\"\/><input type=\"hidden\" name=\"_AccuaForm_buildID\" value=\"accua-form___accua-form__1_69ea045000f29\" id=\"accua-form___accua-form__1_69ea045000f29-element-1\"\/><input type=\"hidden\" name=\"_AccuaForm_wpnonce\" value=\"2d00d6931e\" id=\"accua-form___accua-form__1_69ea045000f29-element-2\"\/><input type=\"hidden\" name=\"_AccuaForm_jsuuid\" id=\"accua-form___accua-form__1_69ea045000f29-element-3\"\/><input type=\"hidden\" name=\"_AccuaForm_referrer\" id=\"accua-form___accua-form__1_69ea045000f29-element-4\"\/><input type=\"hidden\" name=\"_AccuaForm_user_agent\" id=\"accua-form___accua-form__1_69ea045000f29-element-5\"\/><input type=\"hidden\" name=\"_AccuaForm_platform\" id=\"accua-form___accua-form__1_69ea045000f29-element-6\"\/><input type=\"hidden\" name=\"_AccuaForm_tentatives\" id=\"accua-form___accua-form__1_69ea045000f29-element-7\"\/><input type=\"hidden\" name=\"_AccuaForm_submit_method\" value=\"normal\" id=\"accua-form___accua-form__1_69ea045000f29-element-8\"\/><input type=\"hidden\" name=\"_AccuaForm_hash\" id=\"accua-form___accua-form__1_69ea045000f29-element-9\" value=\"qyeWH4bKyvHDtzl\/NJI1NPXMdjg=\"\/><input type=\"hidden\" name=\"_AccuaForm_iv\" id=\"accua-form___accua-form__1_69ea045000f29-element-10\" value=\"fElVWzFrWXk3UHdrJX4vMkZ5UTBlfSpdYE1LR1RZbj12bmQqZTdnMF1uOD4rPCRVLzw7VSkxWkAqYWFwT05VQg==\"\/><input type=\"hidden\" name=\"_AccuaForm_data\" id=\"accua-form___accua-form__1_69ea045000f29-element-11\" value=\"kD6A3IjLpkVbR8hp5aChv5nPDKTY2F+c3tZcZ57wwxrqZR232KF9akAvPx2vxdYCi84+l44S1+oy1GSZrPhqfBCf4O7VlU6Cj51TlfQLzVIB0jEQ5rIk7SfP+TH98B78Ptz22+DbCAXn67jamkk0sbgT+95WlBlG7pLjkTy0gtiwUTp779fXQDk5YDDxdZF+rxlifuI6e6\/aM8Oj2TFY8AkOO2BXdQmw8yVw0FnkweAcQCYjoMfuFL0sFXeeni5xrH9B4agXVimx+L\/9dgqI5O6NfZDjq2FzMhn380Cqrn\/Rq6mwID7wsF+zRBtgCRWsgHVZ8mGtVUYAg7toPQTll7xKjm3RjvjqeK4A22IhWNcNwF1lzObrBMTxO2BbO8yZNS7gk1D0Qb2aCHpZRP8XuKEkMp+HozpxrdtUk2q8k0e5XO1lPoOM5zhTO85m\/6yr5mXB1vVHJy\/\/UFT2oG8G\/oZB0C8oQvX+cKk5RDQYXwttcIUNp5fBrjn8YhuTKKeUdpTRbZMa1gT\/GvRp0GGtB6d7fhjTNInL2nbgYVD268avstJkJu16X2xWpTnDrNAZoBRkFBLPO+3JCTQbl0hzNJwCFQF9pHdNXWk8Iv0XQOAjEpEempdhMp4qBhn+DMi\/L39Tj9x6hCah5mGDd3CXalB1psCM+pBPnPW2cyG+CFenP621CdFPmFg8LiXH4\/BP6C8ydep863rji6pRuI5LoCsrzPJ70+h3R8Zb307WO+wHO4hdgexqEp88naaQxz6TbvZ+Sm38DkycYsXRNU5kXtL\/mO6e4hrhkOAECrG+T4QTacA4K4g8qPI4ZQpYnOyJPdqgswerRX9cUl4ClUU4Z2z2PgihJH7yS2ueMKBS0ODSTppt0rqexiu0FUvMSxb8\/ZlY00YnSgM65W73CP86FvEZa\/neoF4\/sxCZfh66RlZx48U\/Wu3YnIiiA3kLc5ailCBhTCvuLr6gTEWMtqHpvv8RHST2PQfgN81z\/jQtT5mHSlJRB9dV+BD10EJJHWu5diYJ1E6kSKfgQWj0Np4DxMqtX3EK93SxCARpig4MQwzV1F4+xj+BPV4kFhb1H1aPvAfUFJgrAaHa6D8SiqLovymcc+5otJZIWLD3KYCjRTNfsiaBykKznRr00BvQSBq5bXkmook8jJ0gw5MEMBwrAsQcVDrKumIsCSlIAchVHcjIVzpiSAEQXCRg3UKFxDNEVfPzP3l8Baf07T52OVvLpYLkZX5ZkZsnsoBcdUL8ikLbwyhBiedTzsaKI2aKeQrbO7u0KkcfxOe4wYXYLloF7ABdvSyx5Tjfk4B8\/e6rQEpeuG\/kOPo53+YWQ6\/\/O77o8Ua9q9uYP0Zbd9Ms3TPlAW19tbFbe6XLW7YnMEQXsyQioppNBXO1zFtEB96LZrJ9a0ZTxMgV7OvtXdOnLscsR+1huUjVTrXuGsNA0s1CcCIGya2LF3FooK6Gqo\/0hfwF4VrC3esTDonbnWhR8xG7HpL43wuYuKQfmt7Po0Aiks98EzOGYwzcL6mISGkjYDQ7X5A9Sqv\/HjcovLS+mED7JVvEfc0CR+6g8KN6OcAPEQ7TAmHwIhevhJiz52HbPD1SLGPkowX2lC8trnaTNu6BelpdiMiUyQWUK+0pTaJYiK82lCaIeoCXaaI6UPBQIu76gBWpIZXpa1Kqq9\/eUZYLq2S+RKtyIdGkx9L7mnXn7ee557dSpD7QG+sjidFW5spKRPhI45NVgH1gBZwfiRGUNPUyGnIeAh2g3ghCfl2Ln4oylR36M5ZscU9MqGqgI1o\/qbJiqQIbxdNppsBHuFdrtP4BfWUELhr84b3gBD4ujU00XAFplnBmfqybSVFXsSyZjnRU5ow4y0LwQG6+hgtZL5RKapKQ2q0WJCIpcqwkdpLB2H3sb5MRpHE3vWbCj7ARviTeiD3kQlmwxmfiE8\/uthFBveQpC240vZPCay9hzt9LLpWeNS2dvawi4qFAZ6i+BJH13rlz7uwAtSGNVVh+5So46qQS2EI\/V1fjAK6w36IrRBQ9fEw+FgaET0BLwi\/4hq+Wf3nn0iz1mWr22QA5LGBCWyvQMiXjG3+w1KGFxG91h9iV7IWSVfkVj7s59WJ15SAHQl+2165YSky0ItqKK1nuQe4U4zAlmRRu0NJFhR1Yht7dA7Gj+m2i\/VQirpGVkf2ZBsIlfdG3Yi9INMv9iNdDJYiCSI7yvj4lBxcuUICSde\/K\/17DP0qn8tdifWyeuimh9oPRgh0KqPzs+jepnEmtzSznDK9J2O\/YSuOt2VGMQrLln1QbiRh03keQ0pImywFlaVxPpiCtGUx4CQhhxZLU8TqMpVhni6rCDllVXe9OtxUcIyB+jdpVoV2ZBL0bCEhq2bay0kuJ0F8QeokWL+mKf+qmWMoP2ITuMAn9m45n+JqZd8\/wW81JxydYlZrd5OTqXREmnM249sqMXrwnNnpBU0DeKTWiPx0aOPtLGqS4A24fkMeSHr0pyJ1P6NC0m9ToQw8xuECfmn8ynS7DSxbThlVlNPH8x0RwQSR76C7FB+WyfCz6DvkKwo0jrfWVdv44ZZJZO\/qQF0cynHr8oPFmCXtVHOdZLt5vuIFfKxL4VZ7oDhE7s0a4HPNDHD9uNDFE06SPGkgoLrJMMYtnco6BCP\/GYr+nnkBsCTWP5i878i9V+KIaXjlU6zcn7vI9qpk4dQjSZ5LlWlIkpnYb7Cmep5nj3ytpKJknn6fX4IaglYzFOgNmWc5hltuAiUcDwEAdGivA9x8Wr+rgjFT+ntdTRHPNipwK802qxtG2Y9iNVFSLca1xxQPbOvzuyNs2DpgfEZbgT+snmW7Kf2E9f101Xh5NikcKDHST+BXkhmWrJhRy3TRG4qPvmUYW4KqxT\/Eb98yNDs8HVwLzVObUGdLmpJ4yauuL7LWAmh5wIh5GsZ4GmJHoiKihhGt9yiGMDiaXvpK27zqZDEY6CGu2aCBCB6wmJf\/u4VeGPyn\/vAeWs\/NzNI5iiIceJ226ct3mIMRjVR57CgbXTcqmIIYOPyQ838E42FT1QKX0Z+cpfzP2majmqyxMOBsGd2Qaska0K67VG5rlHLIJbL5lXtRr5ID6OsF7zRLwrvj\/WYc7OMWPuhfPaUOoCUl\/m+4IUQRYNtVzbPFZ7FkNZfwgbZDTcKUc1y26Dihkd6GkzSpAgFz5S\/tqW70TOaSq4ePv0mWmUZV9WX+JKZKsbd3Zqi6S3A9OpZGZ0aJKD\/xIKttTcmxiRoERL9ycQ6vc7vMpRC3Egui9vRL90xoIgGZq7uos3JcOuqcU\/i+tuR+FUzAIgJylVKlIKyQ2cz1IUQBI9YuASAfejGdrWC2beHABceEij7F2xot+q8CuSMIImGXCxOSB9g9lNLjrsPZ0H+w2kCtVxRCrAX20oKLUlT03+cg75qUpoD2xkKzS4OKzNbf3VVHX8GP16yXab9UF2Hzt0uK\/ASZ3y79bC3X0I52TSxjY4NjB18uk99ayQXLzynVF5tB2Xjezp+UDqDV++y\/hIoi3ZR4lqBKvL\/yMvTfPgJpGXisZfzBnkecOG74CJHjV2g1MOWpz2UeywcXhM9ZXOdRy6234DzzV01KRCBnC8T6+Nkm7bclwihQKh5znabVOwWCQEwo5avVHJmeQI\/qrgoRF2UHbvKXoDZ6ilbXEhrBCvc8yAdCTj+NxRLanAEbCJ0jJ57NjqooGA1ymVGpEeIcvvJcmQ01VahumAK3ocgp9\/aZCowetkevOf6n22WivD3xV9KfpZ6JFwpv2sHsFQa1WJnNi5lOBRPV3OG8IpCc0+PcIQiaEzki4tUlsaHMp4l9BNIPd9FLOMRkWDM3o2uVBYZtk1ccv0KrnBH2TxQ18bWc7RhBAHVerTOo1p8qsl\/rOe3zNvcEZZSqE1+Cz7wvXAXJdr0SUYlgry8\/bD48V9DJa5EdMDS6kezjOdAN5MC9mnPDqB7VdhFMufFglOViq9IQ7dJroUNsmLTorJtsssXHEaQ47\/6SaZrmbotAStF7hO\/bgjFJzPpSgIgntZVZt8Xaiynq+4z+MS7GfIQh3Hx173+xJL\/kcoz2n+i\/b3K0K8LqUKLAFevv\/NIKXOnlE2Pm6Ho2KJTi0PtJW6T2Zsg8Cp\/olKUy\/lSPCwBIK\/EBp0UPcYRspjpk+0VX0X9R+YjjSfa\/ECwpKTjmiV9Tea69WQoM2p4wVx9hBFUW7ZVk+e1vr\/x4k4Jdeej2NPtHjfOtoXjSEXqjVKTSm+GsbPQpTJcVdp39lVmXJhDSj+G4R1tx6J+MGYQuxF8iV0ze8kP82B5jUr1t1OqC7N9R\/dTxJOV0HHCBOFLMyzBSk6htQdiOnJCheKkZ24yklZGpMpj101oZeuId\/Y+Bl8MgRyV4cxXYDsrnI1THJP7WPc0een\/uXPFCES88WoBZIW9Qi\/ObQ9UO1Vo4f\/gIuzQq7w68zcqw419AjRwh0CXWq5EXMYn1K2i053SpAgOWR5SavkWv\/D3gkow0b+fcHhYslWV+jDFnX70L8bu34G+dxlY19gQMZT0CvivOcZy0Yr3x9wMc7Ec7LqfLy\/aZ\/j56Nf12pbONhruJEdsMxBtDER8+U04Tyc3JRzM5fXgb7VSH3qTfcUMMDtGXw1\/WtlutxibUf9vJMvrzAVwOEWeYkrHQlqfS0tLf4vryb1roqqduUaPeQGIZ1P5\/KIFdd1+aGr4bbIJvCEj6O+ngDA0P1bloFtREVm7BubXVxACajqOTFzdsxvaAau1uSwlGxbizjeXbXOb0O0axna\/QN7KOGMSAcpQF7FTI4Ja7fh6aH0nes2IdjJ\/BypalsoAklb5aQ+kWkBiahC9T7yeAvmoeMpWCiAwWGyvYYf16WdhXQBAGQ\/ZkPNWKOdAwkZacK0zyAKCZJYJfuJUsjTxadrPJtSDPcmszzFeKeTRH+YJWSdNkU\/qhC9OlEJ9vTKHxfDtIcJcxV7jgxQx+Y5D7vQ1eH+Em9zwbs8nejQ8wFutU+11zFrYdKIGL9bFSjkAuwj9soWWUW8Re7xZ8lEQg+7U7J+9ZSDAP6iOmbIf4dw1wU2OVlNtEI5z9ZnBPd5x\/kXpiL+c6BEZpqwN462FCYVuEUkAp+QabfsIo8SMypscjYiD1V8+eqfLTqy2HkVQR+vrNWNSGL39cyIcshmi11JFaZ48j1sfw1jO\/XSnQlCeWMWl+sjRTU0FFV0SIhb6\/q29MFtfRjcnQg9aVKuQwwlfT8qZ7arJMzcAos\/KT4bBLGUq8lWJot8OL5VE6ebnKcnIsjpLQMER0wWCihW5b7iT22TbCMlvck5zh1MxTr9qEFNEyJXMg++iLahTK8IZBKOcC4IRDQOJLNJNg2mfEF4CdoNMPvXTaUYeKWTz3+W2GmHsG1Ut06DfxuX6gxMWwA+rySG\/MNnr352+LFSLooakSr5UihVXZxkVl1vD8hS5V2jEjWwU\/hyiUSuU7bocbvB+HJjksGWbf\/+xhOgu\/tqhEUmib\/2lp5kCu02VevlmCp5eLr64FEhyMtY3tlszIYBS0GCDq6GfL3o7orcbespzC7pRC1pbIAF9EsxV7CdFPqih6+4buZLQpJxa1e7y41jMiOPpp7r2wYiUUjH6kjlTqrUaCmgoNqZRgvWJYbtwpKZhQAMDR5qAIYkgN8OnDgxmMyrSrlnxsCq0jYFkMTd4Bv+ZXNkod1tAj4j9QhNNdmwdFsnUhmrAYODlVYXUJBQlM8tWa2y4+lobG7zz5ijbETB8+u29vWmqbT4ndrA1sk+8dfP08RvyZ4MlIuntL+pYkIfEABJiIxhmrQg9HzGooktu33Y21thOLehVF73UMSltZTsaGzzRZt+RdWhKjr4X+f4cIXfy25bWSGRydWmoi6m\/+0VDK10vw5lZaW0iOBIzZkheHeBSVgp8v6op3VJYt5eyQ3SYoGl0FSWOKV0w9dA+zxU1cphcbxpB8ThADiOHBe\/Q5ZQun+osTeiTPqk818smffKE8mQTlQhfdb8HmYQj0Bof6CsW+KDXPw6BwH\/Pd4VWR1q9Trx6Pc6jSQs21WI8g8WrxLdlIRsAzy8Dib+8I5L3vEvCypgF4f7tmSVFoTF8WdvBY\/8+qfgQ2OBI8zSai4J082gxIZ0t6NYT3DeMwJn6XU3IGIOLlyiP7+BeYQQhFwOdGK0Pemzptring1bOggcCFRy3czDmgY5jmQrrJVRNsi05wVIUoslVDSuQTRUyMXa54yMTR8y010Ok8t\/dt9\/i+\/N7aoC5x83qFw4FBj4jAXViWXV5EZHR+0OQ8\/m8ceUMU+MO9meSR7Z4LjGRZRWpZuV87qkx1SnVYAN6ziUwl\/efNvZPXscBHQPTtlooBS+Hu\/nJcAmKTUfiscVXyNYBjDHIZ2K9oiTpX8NExI9HjsaApaZ9QMilW+3mQw8D4jn\/a\/JRIIUNmMN2D+n2jCHghpMVMTTPGh9EmkJ9peDNyfTTOeVz7fBjGiMvNQCAfrzFATazakayeKXiZRTDJ6VTZ1eO99aiCD0nnbmt7s9EDq4tx99JE3885suUZWQSqEZ+4ltp1GRAe43csiCTMn51EvVR7HycakYaDh9PO551Yj3\/2OzSeTHmDYMojRntfuY9ZVo0a4QtruqGt5EOC9H3gNMyNyQDUyazw8zxKl70RZ+BZQtgjod30r5M3oYNf7ydKjxy+qZ0F6Nbj0MQSZajrPimKn+rBxughnpTxrM3lEqg2r1K1A2qgYQ6kslUf7nro\/f6cRxQjEYO3gMH5HP\/uVXEnkjYG8WEIT0+R2MUsvzQC\/7YO18ij8FHLDF7isoWBtTk363jII1F3\/CfyMe9oLY3MbZ+Kt7M+7NXpIUTWA89lXMtzYDKtFU9TTC\/r\/CDyo4N+Hfh2vuSe+LAB1YthqhsYMDRY333yMyTddm1RyW6xzhveORWtUgF1Pt20Yvoag06ynGjAHWsFG5DFqzF+WSIpjDax5hmxnI2jKf\/iMOD2H0FGZI2bNSbyVKWpzAsQbhE9zojbE88qS1mfhi0La0q+5knoYt4M+fD\/0I\/kS6ojdFhZSKFa7Kt5wIU4UWMHTsjaEjEfp37\/gOa31p3qZrlo9K9VH87Nv55q10X5QVCUf9tpeLtTSdau8+wv3GDv75EJzDlNFdAfgbjajzuLPXBft9yFbztO4\/hm3Ba\/KxdS8AJ0B4A\/VzINwGASri42pxqjUELx5XYoII83UBOrhTyr0oYMKF6XEdXkmHww2aU1Rt\/j7gsxo\/Hd72Q9qx5rawG1lZgMs8aWAgTbYMXdpogJLrH+Ge2BMT\/CqtB6sspICs58pbIc7buCI+gdNVKoQ98rxReS1\/8Q6rSu3lqQLgznauBEwESM6Vs7upq5bSbZC8rrV8hbSKnvVNwcuFUWtlqtbZRT2Lc0KiY3y9RcPSDJgZBld9NxNFADBBt9J7HpjViPxuHCBVsxy2ELFql+eV08QL914p0M9g66tRxZafkZXl3sQrU1daXaSgezACU583Uy12js3xoG\/3gpX+F\/DZIegZYS0yPS5rXexQLBLecpkR7KgGwPJghThTY4y7OMd2u4I7TwNriIIbT6FGISaBaa8fqWhFxDodbR6D1TUe8B1ryhB3ajg8SjNy3GvoBjVFIw4CO+bn\/ilQXzrIZO8AyfzrQFayD1+jovrytxMIucWP48c8S+F20IWvB6CxgcSvxlpgcjzMvPFp4Dboamuo1EkUJhluvUL7YtmDEiBxOIXXQr9bdaUOBRstHe7XCQXv+vzIVIvd+PVrsRvKCF5s2T8HGchgcgwAAobCe69y+xVznsD57vuWu484Vuvsvsp3uyZsFe3Ei\/IotlVBTiXdt8QKbnIgODow+K3BTueZOiKcXAr9JFkrmX+Vihshc9xEWnVWkhetsvIaa0JUTP4neP4gvd4oyKv1w477EnazPPk74C2UYeuHtQUX56SgIczB7naOVtjK9vbk6lDmeI70bj1c7fHKev6OL28Ib9\/r3nOI4YvExAT7z\/hYAn7zNVjLzbk9noJKEDB2dg2gWYD5ld2XV7TPfZGalAKR00JuxgAU0MyhXZtk\/VnakW8NBRC5kHv87FV7EocnH43hbgPJkffUqAmrtfvLEp6VoLENHp3efjNiiAJHpy3K8gUbkLq+YnskNpFi8FG1UCUdcKVy7CLnKTJs\/VAia2H96qnbThs8XadLCrJBaHvH69n2zriFFIRVlfLIo49X6vthUBj3olZOrV8dG6Z3x7Z\/IVSI8LHUOAvgU+MjDSOmJ9F0SsTBDU1RHdBJECaIC\/+kACWT7x0\/vv4RR33uq2fC2C+gVind2UIbbHZcgvqBRXmHUJ3NmmgyiBVEyJB1M8yiwPlbqdpBf5Q\/Y9ZAeSB1uofydGZ+zCmi2cc5rGQZ76EoeVwLHm7o\/SITlWbtoLGU9HfDrDF+n7eGouM1dUlZODGrcB\/QKYFwF7z5Yq1MSrxuUGAH6sZ9Dk4\/qp\/tyeDEKJLW4dqDLNKi+3oQC+5oXPbbC199GxWHe9vzHTq0vaEN3yUFIhjKYmW\/ZEl1Df9n29E1CgKarchDhBzNE2EaQzqaIUjYh3t82e9SSaN6699EvgMPr5GrHYjueENcQvI729u21+9\/JbVvu91wkOKF+P16HO3PSJJc3tdZML+ySdRxQ1W9bE9E4Ln5o6YWTEkEpwZ5MGosX4hIZvx5UnWYpbr64a5mnX4ikb15OoTGHoZ1se2QqNeRS2ytLUt7cs5BamDZiXwEz\/tPia+Ui7IYck8ctx\/rCcxZ8Z8uTNameL4D3TOUuQSYsU\/NqvafKn2WlrU\/FaKERFGurERRYaLvn3L2USgnXZ\/M7a36YQ7xrhr77+AUAwo6fJaQ+ieTenYzLxAb5bvYKQiVFgOXYhQf4hQ\/R8Y\/M5vGcHeK0SLLgczUbBfOGPtgJ\/Pbo28ua4uWFtT8EYSn+GT9LXt+eCNFA2i3mPMJEAlQ0FiUH81E10\/WoiLJyUUIIN2cgxPiqLcFT14VQsMaka0FkI4jKpHRmOERDKeEHu4xtOZ4L2lQyPn6ivvgacc6Wcj2eaN\/aE\/U3UDlZbgMeTffLpc31svO0XA0UQWS+C4JSxM\/5IjUY5VBPYtiVBTBGGFQNcH78RWC5KgKYxGs0IKUzaxAEKlo7J\/enm+D0oKeUOmcaGl5hYQ3SgV1tLj3axkcukHEIs03hvQEfOTWwvbgUsHpZwkvfrpvinFAmltvoB7qDJi+mScg6px85RQ6e\/obVo1AhQAppiwOp+6xV1z8zIm8gTcLqcxtNHDKo5ov9RJ1QtjBL5h9YOduYshQBZWxsiK2WpbVDOrmsEvGYHu9C7WkPwvIGtjCYKLaT1qv8va63OSB3zvq8cDplj9b6WHS5Id8E+oCJhofqKORGYWPu297F6mOWQ0RXCnKXE8RWZzQc+ZroHga\/lZAEiAe+5+nJlXCcUqAdTnagiBey4txevlwTP03g7KF3OeuBtitP\/A3i9MA4nL3MuWSY0WcRWqNxds08ySBdXT5481YK+WkvxT\/t57DdX7XIiVA3MWLOiUW3WBHcqkC8F5b6T\/rn5qYZ1w9\/sdIwtOMK1tB3Dkvqd1RwFn1+Id0Lm\/o4h9JhJZUG2aJbsh60Km4PIx4Jy4RCghmfC7JnuVhnvoDZkyYtZYEJELFEeg+hKJ4e8\/mfpujgSeaIX6vF+zaQYvFli\/N8V\/BB5am7opYzNmt7+5XUIoUY71kpo36lf7adEvFSQdS2Tl2\/3aO8WPB\/waFTTietNIBTtNTyuu\/Rsyw4UhRkukyHFtdo55dS\/OAipj1eN1oKrIuShHvWBSeTltd57ZSKh+RO46ZDZSwcs1H6ygJ9OzVsXpYyjq6dDThWsW9lMeqPnfS2lMotbk\/RwBM6yVS2MTjQdyDXJxLSyCblk+2vQt1yxytxawM0flaBG72+U3y0P5GSPIEbbYwdC8bex5QeLhtLNDKjNlzOmwfhbRbAcluFg3HUpVssAJJV911I1Td02XDoZsm+rnx935BEuOks9+YZyp5BcDbOqtkd4y1cN0o\/shq8K5HNGt2wlAtzRN\/1FPj2lz0nYfGOEVWqJm0ZAraqSfjnBhnxBl1JiI8HZqnuKlEqsuyRmFLm90iTRHDOxSvQQTmatf3OWW5+X+rz4ib+4ZNJkmjy3GKcoEvsyTXSSMtwnmnalJe4YGmS3rgAnYclHbbAhA\/c6Sf9cGpX558l+hFYa\/+6kFBXZoI3ACKPFsMmx8tYjk9dzN82iY+nN6ircSidoOBpVHtu+m4VkAEmZhUG2o1XM2\/YIQVnIAYe1HoGDhVdj\/elvIma8rPO2yzUtMOW2T4\/Mfdj7KtNYDu5QKLOZ9K5bQ38CPW6yJ+4Vy4MMuuEYfcdmE82Yvpz7po0F8OFoax8fUPUfq0W4acitaShle6327hzBcPzVnpyMaOQ2y8PImeHD29q2nuFgBAfBNc2fmhfgNpxo\/t0lu5d0YmYAg+KdjJmgW3FYP8ITCNXmZFECuwvisUKZqhpTJA4NghIbMSAOUk6p7femKaV5bQ4tdnBJlY496uTanWFnXEz8ZP+8haTuj4Xg\/tVzuCtft7fzwYPRtAkR0h7VWax\/e6nk554dBVV\/cpY6dPCxubwUX6vQrKx2FC2Z0DFjKvdFIn9bpEQ7G9T\/wErLgs477i\/XHJ4M9qTJ5Q0nVJXuE0WcTSJzjBbUH9jdCd3OrTbMpY\/TSe3Qc5B5dyIj1nW2Xq4kRgc1k8A9wlEZPBEwOGG9lIPbjiam1R74VzU5x061F5Rh2nXmvxQ7kKgw1bSguiAf7G+4cOlAScccKVNUPHsdUqy3blcwHN4HdOQ9tw3xM7RE6bfKOtB3uyeC+6rZziFj\/acAoMQe9DVk9R1t8hltcoN12LHZ5bQ+8bz6Rx66Lb\/iaxbB3Q7cIXUPCoXmRsHF\/D1VKewWaKBj6CbQLgf2LK3VYHenN+IiCa0UKFbgb\/lvMXZ4AwDwKKMOrqb2OTZ3POVRB0PHeRogEppeQfSozK2YpnffgtYwc17FKX6AxW+ikECpOMLB5Ff5yXUGqNa4JXAY9FALFCroTA39DO04WIQUDz0JQVb2aWEAWvsFKS\/Eilq7cykk8osYBS\/r9lVUqUf\/zQ6XRxLWF+v7eCWYuwVGrAWX4aaG97D1NGNH8RJoJh8RZX3hvRG+12ak4Hp9VBlzBJUqQDdf1w48\/qVMKD7kFI40HEBQYY0QhOR7vi9VAnUTx\/6tZDUX43mD9r9Teyf8SF23UDGVBVSkiRwIkXnEMoVb15+1HZW6EqkNFEjqd1yrem1rd4cjhW1+Lgq\/PWAfFHO1+FadEmyy8DFcNL5l789EXDPccXhzr4QSRHc38BwFQBZ8le4cst\/E7EEoIj\/JDUTUQrs9HiqzsD6ho0g9ymasqPLGOb3IXvBzhUqVvkIfXL6Gs2GoAZWjIdaYl9Ivyq66+TTCh3mJcZku3ghB6EFKqn1nMjqq6UN+bVjxyvp1mipU5EXjo\"\/><div class=\"pfbc-element-0 pfbc-element\"><div class=\"pfbc-label\"><label for=\"accua-form___accua-form__1_69ea045000f29-element-12\">Etunimi <span class=\"pfbc-required\" aria-label=\"required\">*<\/span><\/label><\/div><div class=\"pfbc-fieldwrap\"><input type=\"text\" class=\"pfbc-textbox accuaforms-field-required accuaform-fieldname-first_name accuaform-fieldtype-textfield\" name=\"first_name\" value=\"\" id=\"accua-form___accua-form__1_69ea045000f29-element-12\" required=\"required\" aria-required=\"true\"\/><\/div><div class=\"pfbc-elementbottom\"><\/div><\/div><div class=\"pfbc-element-1 pfbc-element\"><div class=\"pfbc-label\"><label for=\"accua-form___accua-form__1_69ea045000f29-element-13\">Sukunimi <span class=\"pfbc-required\" aria-label=\"required\">*<\/span><\/label><\/div><div class=\"pfbc-fieldwrap\"><input type=\"text\" class=\"pfbc-textbox accuaforms-field-required accuaform-fieldname-last_name accuaform-fieldtype-textfield\" name=\"last_name\" value=\"\" id=\"accua-form___accua-form__1_69ea045000f29-element-13\" required=\"required\" aria-required=\"true\"\/><\/div><div class=\"pfbc-elementbottom\"><\/div><\/div><div class=\"pfbc-element-2 pfbc-element\"><div class=\"pfbc-label\"><label for=\"accua-form___accua-form__1_69ea045000f29-element-14\">S\u00e4hk\u00f6postiosoite <span class=\"pfbc-required\" aria-label=\"required\">*<\/span><\/label><\/div><div class=\"pfbc-fieldwrap\"><input type=\"email\" class=\"pfbc-textbox accuaforms-field-required accuaform-fieldname-email accuaform-fieldtype-autoreply_email\" name=\"email\" value=\"\" id=\"accua-form___accua-form__1_69ea045000f29-element-14\" required=\"required\" aria-required=\"true\"\/><\/div><div class=\"pfbc-elementbottom\"><\/div><\/div><div class=\"pfbc-element-3 pfbc-element\"><div class=\"pfbc-label\"><label for=\"accua-form___accua-form__1_69ea045000f29-element-15\">Puhelinnumero<\/label><\/div><div class=\"pfbc-fieldwrap\"><input type=\"text\" class=\"pfbc-textbox accuaform-fieldname-puhelinnumero accuaform-fieldtype-textfield\" name=\"puhelinnumero\" value=\"\" id=\"accua-form___accua-form__1_69ea045000f29-element-15\"\/><\/div><div class=\"pfbc-elementbottom\"><\/div><\/div><div class=\"pfbc-element-4 pfbc-element\"><div class=\"pfbc-label\"><label for=\"accua-form___accua-form__1_69ea045000f29-element-16\">Yritys<\/label><\/div><div class=\"pfbc-fieldwrap\"><input type=\"text\" class=\"pfbc-textbox accuaform-fieldname-yritys accuaform-fieldtype-textfield\" name=\"yritys\" value=\"\" id=\"accua-form___accua-form__1_69ea045000f29-element-16\"\/><\/div><div class=\"pfbc-elementbottom\"><\/div><\/div><div class=\"pfbc-element-5 pfbc-element\"><div class=\"pfbc-label\"><label for=\"accua-form___accua-form__1_69ea045000f29-element-17\">Viesti <span class=\"pfbc-required\" aria-label=\"required\">*<\/span><\/label><\/div><div class=\"pfbc-fieldwrap\"><textarea class=\"pfbc-textarea accuaforms-field-required accuaform-fieldname-message accuaform-fieldtype-textarea\" rows=\"5\" name=\"message\" cols=\"50\" id=\"accua-form___accua-form__1_69ea045000f29-element-17\" required=\"required\" aria-required=\"true\"><\/textarea><\/div><div class=\"pfbc-elementbottom\"><\/div><\/div><div class=\"pfbc-element-6 pfbc-element\">\n\n<!-- ReCaptcha 1 is discontinued, please go to Contact Forms settings page and set reCaptcha v2 keys -->\n\n<div class=\"pfbc-elementbottom\"><\/div><\/div><div class=\"pfbc-element pfbc-buttons\"><button type=\"submit\" value=\"L\u00e4het\u00e4\" name=\"laheta\" class=\"accuaform-fieldname-laheta accuaform-fieldtype-submit\" id=\"accua-form___accua-form__1_69ea045000f29-element-19\">L\u00e4het\u00e4<\/button><\/div><\/form><script type=\"text\/javascript\">\r\n<!-- \r\njQuery(document).ready(function() {    \r\n    \/\/ If there are errors on page load, focus the error summary or first invalid field\r\n    if (jQuery('.pfbc-error', '#accua-form___accua-form__1_69ea045000f29').length > 0) {\r\n      setTimeout(function() {\r\n        var errorContainer = jQuery('.pfbc-error', '#accua-form___accua-form__1_69ea045000f29').first();\r\n        errorContainer.attr('tabindex', '-1').focus();\r\n      }, 100);\r\n    } else if (jQuery('.pfbc-element-has-error', '#accua-form___accua-form__1_69ea045000f29').length > 0) {\r\n      setTimeout(function() {\r\n        var firstInvalidField = jQuery('.pfbc-element-has-error :input:visible:enabled:first', '#accua-form___accua-form__1_69ea045000f29').first();\r\n        if (firstInvalidField.length) {\r\n          firstInvalidField.focus();\r\n        }\r\n      }, 100);\r\n    }\r\n    jQuery(\"#accua-form___accua-form__1_69ea045000f29 .pfbc-element:last\").css({ \"margin-bottom\": \"0\", \"padding-bottom\": \"0\", \"border-bottom\": \"none\" });\t});\r\n\/\/ -->\r\n<\/script><script type=\"text\/javascript\">\r\n<!--\r\nvar _handle_ajax_submit_accua_form___accua_form__1_69ea045000f29 = function() {return true;}\r\nvar _handle_ajax_submit_complete_accua_form___accua_form__1_69ea045000f29 = function() {return false;}\r\nvar _handle_ajax_submit_timeout_accua_form___accua_form__1_69ea045000f29 = function() {return false;}\r\nvar _handle_ajax_submit_message_accua_form___accua_form__1_69ea045000f29 = function() {}\r\nvar _handle_ajax_submit_response_accua_form___accua_form__1_69ea045000f29 = function() {}\r\n\r\njQuery(function($) {\r\n  var thisform = $(\"#accua-form___accua-form__1_69ea045000f29\");\r\n  var ajax_enabled = \"www.perustele.fi\" == location.hostname ;\r\n  var anchor_id = \"1\" ;\r\n\r\n  var response_messages = $(\"#_response_messages_accua-form___accua-form__1_69ea045000f29\");\r\n  if (! response_messages.length) {\r\n    \/\/ Create all three anchors for success, invalid, and error states\r\n    thisform.before('\\x3Ca id=\"formSubmitSuccess-'+anchor_id+'\" name=\"formSubmitSuccess-'+anchor_id+'\" class=\"accua-form-anchor\" \/\\x3E');\r\n    thisform.before('\\x3Ca id=\"formSubmitInvalid-'+anchor_id+'\" name=\"formSubmitInvalid-'+anchor_id+'\" class=\"accua-form-anchor\" \/\\x3E');\r\n    thisform.before('\\x3Ca id=\"formSubmitError-'+anchor_id+'\" name=\"formSubmitError-'+anchor_id+'\" class=\"accua-form-anchor\" \/\\x3E');\r\n    response_messages = $('\\x3Cdiv id=\"_response_messages_accua-form___accua-form__1_69ea045000f29\" class=\"accua-form-messages\"\\x3E\\x3C\/div\\x3E');\r\n    thisform.before(response_messages);\r\n  }\r\n  \r\n  \/\/ Smooth scroll helper function - also updates URL hash for GA tracking and bookmarkability\r\n  var smoothScrollToElement = function(elementId) {\r\n    var target = document.getElementById(elementId);\r\n    if (target) {\r\n      target.scrollIntoView({ behavior: 'smooth', block: 'start' });\r\n      \/\/ Update URL hash without adding a browser history entry (replaceState prevents instant jump)\r\n      if (history.replaceState) {\r\n        history.replaceState(null, '', '#' + elementId);\r\n      }\r\n    }\r\n  };\r\n\r\n  var _ajax_submitting_accua_form___accua_form__1_69ea045000f29 = false;\r\n  var timeout_handler = false;\r\n  var timeout_count = 0;\r\n  var fail_count = 0;\r\n  var disabled_fields = false;\r\n  var submitBtn = $('button[type=\"submit\"]', thisform);\r\n  var submitBtnOriginalText = submitBtn.text();\r\n  var submitBtnSendingText = \"Sending...\";\r\n\r\n  var jsuuid_field = $('input[name=\"_AccuaForm_jsuuid\"]', thisform);\r\n  var jsuuid = jsuuid_field.val();\r\n  if (jsuuid == '') {\r\n    var chars = '0123456789abcdefghijklmnopqrstuvwxyz'.split('');\r\n    var radix = chars.length\r\n    for (i = 0; i < 25; i++) {\r\n      jsuuid += chars[0 | Math.random()*radix];\r\n    }\r\n    jsuuid_field.val(jsuuid);\r\n  }\r\n\r\n  var ga_track = {\"title\":\"Lis\\u00e4tietoja\",\"track_submit\":false,\"track_fields\":false} ;\r\n  var ga_event, ga_submit_event, ga_field_event, ga_field_events_fired = {};\r\n  ga_event = function(eventCategory, eventAction){\r\n    \/* matomo *\/ \r\n    if (window._mtm && (typeof window._mtm.push == 'function')) {\r\n      window._mtm.push({'event': 'ContactForms', 'eventAction': eventAction, 'eventCategory': eventCategory, 'eventLabel': ga_track.title});\r\n    }\r\n      \r\n    if (typeof window.gtag == 'function') {\r\n      window.gtag('event', eventAction, {'event_category': eventCategory, 'event_label': ga_track.title});\r\n    } else if (window.dataLayer && (typeof window.dataLayer.push == 'function')) {\r\n      dataLayer.push({'event': 'ContactForms', 'eventAction': eventAction, 'eventCategory': eventCategory, 'eventLabel': ga_track.title});\r\n      \/\/backward compatibility\r\n      var gtag = function(){window.dataLayer.push(arguments);}\r\n      gtag('event', eventAction, {'event_category': eventCategory, 'event_label': ga_track.title});\r\n    } else if (typeof window.ga == 'function') {\r\n      window.ga('send', 'event', eventCategory, eventAction, ga_track.title);\r\n    } else if (window._gaq && (typeof window._gaq.push == 'function')) {\r\n      window._gaq.push(['_trackEvent', eventCategory, eventAction, ga_track.title]);\r\n    } else if (window.gaq && (typeof window.gaq.push == 'function')) {\r\n      window.gaq.push(['_trackEvent', eventCategory, eventAction, ga_track.title]);\r\n    } else if (window.pageTracker && (typeof window.pageTracker._trackEvent == 'function')) {\r\n      window.pageTracker._trackEvent(eventCategory, eventAction, ga_track.title);\r\n    }\r\n  }\r\n  ga_submit_event = function(eventAction){\r\n    if (ga_track.track_submit) {\r\n      ga_event('ContactFormsSubmit', eventAction);\r\n    }\r\n    thisform.trigger('ContactFormsSubmit', [eventAction]);\r\n  }\r\n  ga_field_event = function(field_name){\r\n    if (!ga_field_events_fired[field_name]){\r\n      if (ga_track.track_fields) {\r\n        ga_event('ContactFormsFieldFilledIn', field_name);\r\n        ga_field_events_fired[field_name] = true;\r\n      }\r\n      thisform.trigger('ContactFormsFieldFilledIn', [field_name]);\r\n    }\r\n  }\r\n  $('input, textarea, select', thisform).change(function(){\r\n    ga_field_event($(this).attr('name'));\r\n  });\r\n  \r\n  \/\/ Disable submit button during form submission\r\n  var disableSubmitButton = function() {\r\n    submitBtn.prop('disabled', true).attr('aria-busy', 'true').text(submitBtnSendingText);\r\n  };\r\n  \r\n  \/\/ Re-enable submit button (on failure or completion)\r\n  var enableSubmitButton = function() {\r\n    submitBtn.prop('disabled', false).removeAttr('aria-busy').text(submitBtnOriginalText);\r\n  };\r\n  \r\n  \/\/ Get field label from the field's container element\r\n  var getFieldLabel = function(field) {\r\n    var container = field.closest('.pfbc-element');\r\n    if (!container.length) {\r\n      return '';\r\n    }\r\n    \r\n    \/\/ Try floating label first (inline labels mode)\r\n    var label = container.find('.pfbc-floating-label').first();\r\n    if (!label.length) {\r\n      \/\/ Try standard label in .pfbc-label\r\n      label = container.find('.pfbc-label label').first();\r\n    }\r\n    if (!label.length) {\r\n      \/\/ Try any label element\r\n      label = container.find('label').first();\r\n    }\r\n    \r\n    if (label.length) {\r\n      \/\/ Get text content, removing the required asterisk\r\n      var text = label.clone().find('.pfbc-required').remove().end().text().trim();\r\n      return text;\r\n    }\r\n    \r\n    return '';\r\n  };\r\n  \r\n  \/\/ Build required field error message with field name\r\n  \/\/ Checks per-field data-custom-required-msg attribute first, then falls back to global template\r\n  var getRequiredMessage = function(fieldLabel, field) {\r\n    var template = \"%s is a required field\";\r\n    if (field) {\r\n      var custom = field.attr('data-custom-required-msg');\r\n      if (custom) {\r\n        template = custom;\r\n      }\r\n    }\r\n    if (fieldLabel) {\r\n      return template.replace('%s', fieldLabel);\r\n    }\r\n    return template.replace('%s', '');\r\n  };\r\n  \r\n  \/\/ Build email validation error message with field name\r\n  \/\/ Checks per-field data-custom-format-msg attribute first, then falls back to global template\r\n  var getEmailMessage = function(fieldLabel, field) {\r\n    var template = \"%s: please enter a valid email address\";\r\n    if (field) {\r\n      var custom = field.attr('data-custom-format-msg');\r\n      if (custom) {\r\n        template = custom;\r\n      }\r\n    }\r\n    if (fieldLabel) {\r\n      return template.replace('%s', fieldLabel);\r\n    }\r\n    return template.replace('%s', '');\r\n  };\r\n  \r\n  \/\/ Build phone validation error message with field name\r\n  \/\/ Checks per-field data-custom-format-msg attribute first, then falls back to global template\r\n  var getPhoneMessage = function(fieldLabel, field) {\r\n    var template = \"%s: please enter a valid phone number (e.g. +39 333 1234567)\";\r\n    if (field) {\r\n      var custom = field.attr('data-custom-format-msg');\r\n      if (custom) {\r\n        template = custom;\r\n      }\r\n    }\r\n    if (fieldLabel) {\r\n      return template.replace('%s', fieldLabel);\r\n    }\r\n    return template.replace('%s', '');\r\n  };\r\n  \r\n  \/\/ Check if a telephone field contains only a country prefix (e.g. \"+39\")\r\n  \/\/ These values should be treated as empty \u2014 the user hasn't entered a real number.\r\n  var isTelephonePrefixOnly = function(field) {\r\n    if (!field.hasClass('accuaform-telephone')) return false;\r\n    var val = field.val();\r\n    if (!val) return false;\r\n    val = val.trim();\r\n    if (val === '' || val.charAt(0) !== '+') return false;\r\n    return val.replace(\/\\D\/g, '').length <= 4;\r\n  };\r\n  \r\n  \/\/ Scroll to first invalid field, then focus after scroll completes\r\n  var focusFirstInvalidField = function() {\r\n    var firstInvalidContainer = $('#accua-form___accua-form__1_69ea045000f29 .pfbc-element-has-error').first();\r\n    if (!firstInvalidContainer.length) {\r\n      return;\r\n    }\r\n    \r\n    \/\/ Try to find a focusable element in the invalid container\r\n    var focusTarget = null;\r\n    \r\n    \/\/ Check for radio\/checkbox groups first\r\n    var radioOrCheckbox = firstInvalidContainer.find('input[type=\"radio\"], input[type=\"checkbox\"]').first();\r\n    if (radioOrCheckbox.length) {\r\n      focusTarget = radioOrCheckbox;\r\n    }\r\n    \r\n    \/\/ Check for file input (focus the dropzone button if available)\r\n    if (!focusTarget) {\r\n      var fileDropzone = firstInvalidContainer.find('.accua-file-dropzone');\r\n      if (fileDropzone.length) {\r\n        focusTarget = fileDropzone;\r\n      } else {\r\n        var fileInput = firstInvalidContainer.find('input[type=\"file\"]');\r\n        if (fileInput.length) {\r\n          focusTarget = fileInput;\r\n        }\r\n      }\r\n    }\r\n    \r\n    \/\/ Check for custom select buttons (post select, etc.)\r\n    if (!focusTarget) {\r\n      var customSelectBtn = firstInvalidContainer.find('.pfbc-post-select-trigger, button[aria-haspopup=\"listbox\"]').first();\r\n      if (customSelectBtn.length) {\r\n        focusTarget = customSelectBtn;\r\n      }\r\n    }\r\n    \r\n    \/\/ Standard inputs (text, email, select, textarea)\r\n    if (!focusTarget) {\r\n      focusTarget = firstInvalidContainer.find('input:not([type=\"hidden\"]), textarea, select').first();\r\n    }\r\n    \r\n    if (focusTarget && focusTarget.length) {\r\n      \/\/ Scroll into view first with smooth animation\r\n      if (focusTarget[0].scrollIntoView) {\r\n        focusTarget[0].scrollIntoView({ behavior: 'smooth', block: 'center' });\r\n      }\r\n      \/\/ Focus after scroll animation completes (typical scroll animation is ~300-500ms)\r\n      setTimeout(function() {\r\n        focusTarget.focus();\r\n      }, 500);\r\n    }\r\n  };\r\n  \r\n  \/\/ Scroll to a specific field by ID and focus it\r\n  var scrollToFieldAndFocus = function(fieldId) {\r\n    var element = $('#' + fieldId);\r\n    if (!element.length) {\r\n      return;\r\n    }\r\n    \r\n    var focusTarget = element;\r\n    \r\n    \/\/ If it's already an input\/select\/textarea, use it directly\r\n    if (element.is('input, select, textarea')) {\r\n      focusTarget = element;\r\n    }\r\n    \/\/ For containers (pfbc-element), find the first focusable element\r\n    else if (element.hasClass('pfbc-element')) {\r\n      var radioOrCheckbox = element.find('input[type=\"radio\"], input[type=\"checkbox\"]').first();\r\n      if (radioOrCheckbox.length) {\r\n        focusTarget = radioOrCheckbox;\r\n      } else {\r\n        var fileDropzone = element.find('.accua-file-dropzone');\r\n        if (fileDropzone.length) {\r\n          focusTarget = fileDropzone;\r\n        } else {\r\n          focusTarget = element.find('input:not([type=\"hidden\"]), textarea, select').first();\r\n        }\r\n      }\r\n    }\r\n    \/\/ For file dropzone wrapper, focus the dropzone\r\n    else if (element.find('.accua-file-dropzone').length) {\r\n      focusTarget = element.find('.accua-file-dropzone').first();\r\n    }\r\n    \r\n    if (focusTarget && focusTarget.length) {\r\n      if (focusTarget[0].scrollIntoView) {\r\n        focusTarget[0].scrollIntoView({ behavior: 'smooth', block: 'center' });\r\n      }\r\n      setTimeout(function() {\r\n        focusTarget.focus();\r\n      }, 500);\r\n    }\r\n  };\r\n  \r\n  \/\/ Track if form submission has been attempted\r\n  var submitAttempted = false;\r\n  \r\n  \/\/ Array to collect field errors for summary\r\n  var fieldErrorsList = [];\r\n  \r\n  \/\/ Get the error ID for a radio\/checkbox group (DRY helper)\r\n  var getGroupErrorId = function(field) {\r\n    var groupWrapper = field.closest('.pfbc-radio-buttons, .pfbc-checkboxes').parent();\r\n    var groupId = groupWrapper.attr('id') || field.attr('id');\r\n    return groupId + '-error';\r\n  };\r\n  \r\n  \/\/ Remove an error element with smooth animation (generic helper for any field type)\r\n  var removeErrorAnimated = function(errorId) {\r\n    var errorEl = $('#' + errorId);\r\n    if (errorEl.length) {\r\n      errorEl.addClass('pfbc-error-removing');\r\n      setTimeout(function() {\r\n        errorEl.remove();\r\n      }, 150); \/\/ Match CSS animation duration\r\n    }\r\n  };\r\n  \r\n  \/\/ Remove all existing errors for a group with smooth fade-out (handles both potential IDs)\r\n  var removeGroupErrors = function(field, fieldName, animate) {\r\n    var groupContainer = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n    var errors = groupContainer.find('.pfbc-inline-error');\r\n    \r\n    \/\/ Also collect errors by potential IDs\r\n    var groupWrapper = field.closest('.pfbc-radio-buttons, .pfbc-checkboxes').parent();\r\n    var wrapperId = groupWrapper.attr('id');\r\n    if (wrapperId) {\r\n      errors = errors.add($('#' + wrapperId + '-error'));\r\n    }\r\n    var firstField = $(\"[name='\"+fieldName+\"']\", thisform).first();\r\n    if (firstField.attr('id')) {\r\n      errors = errors.add($('#' + firstField.attr('id') + '-error'));\r\n    }\r\n    \r\n    if (animate && errors.length) {\r\n      \/\/ Smooth fade-out animation\r\n      errors.addClass('pfbc-error-removing');\r\n      setTimeout(function() {\r\n        errors.remove();\r\n      }, 150); \/\/ Match CSS transition duration\r\n    } else {\r\n      errors.remove();\r\n    }\r\n  };\r\n  \r\n  \/\/ Show or update the error\/success\/loading summary area\r\n  \/\/ state: true (success), false (error), 'loading' (submitting)\r\n  var updateSummaryArea = function(state) {\r\n    var summaryArea = $('#accua-form___accua-form__1_69ea045000f29-validation-summary');\r\n    \r\n    if (!submitAttempted && state !== 'loading') {\r\n      summaryArea.remove();\r\n      return;\r\n    }\r\n    \r\n    \/\/ Ensure summary area exists\r\n    if (!summaryArea.length) {\r\n      summaryArea = $('<div id=\"accua-form___accua-form__1_69ea045000f29-validation-summary\" class=\"pfbc-validation-summary\" role=\"status\" aria-live=\"polite\"><\/div>');\r\n      thisform.find('.pfbc-error').remove();\r\n      thisform.append(summaryArea);\r\n    }\r\n    \r\n    if (state === 'loading') {\r\n      \/\/ Show loading state - neutral blue with spinner and text\r\n      summaryArea\r\n        .removeClass('pfbc-validation-error pfbc-validation-success')\r\n        .addClass('pfbc-validation-loading')\r\n        .attr('role', 'status')\r\n        .attr('aria-live', 'polite')\r\n        .attr('aria-busy', 'true')\r\n        .html('<span class=\"pfbc-summary-spinner\" aria-hidden=\"true\"><\/span>' + \"Submitting your form, please wait...\");\r\n    } else if (state === true) {\r\n      \/\/ Show success state\r\n      summaryArea\r\n        .removeClass('pfbc-validation-error pfbc-validation-loading')\r\n        .addClass('pfbc-validation-success')\r\n        .attr('role', 'status')\r\n        .attr('aria-live', 'polite')\r\n        .removeAttr('aria-busy')\r\n        .html('<span class=\"pfbc-summary-icon\">\u2713<\/span> ' + \"All fields are valid. Ready to submit!\");\r\n    } else {\r\n      \/\/ Show error state with field list\r\n      summaryArea\r\n        .removeClass('pfbc-validation-success pfbc-validation-loading')\r\n        .addClass('pfbc-validation-error')\r\n        .attr('role', 'alert')\r\n        .attr('aria-live', 'assertive')\r\n        .removeAttr('aria-busy');\r\n      \r\n      var html = '<p class=\"pfbc-summary-header\">' + \"Check the following fields to continue:\" + '<\/p><ul class=\"pfbc-summary-list\">';\r\n      for (var i = 0; i < fieldErrorsList.length; i++) {\r\n        var err = fieldErrorsList[i];\r\n        html += '<li><a href=\"#\" class=\"pfbc-summary-link\" data-field-id=\"' + err.fieldId + '\">' + err.label + '<\/a> \u2013 ' + err.errorType + '<\/li>';\r\n      }\r\n      html += '<\/ul>';\r\n      summaryArea.html(html);\r\n      \r\n      \/\/ Attach click handlers to links\r\n      summaryArea.find('.pfbc-summary-link').on('click', function(e) {\r\n        e.preventDefault();\r\n        var fieldId = $(this).data('field-id');\r\n        scrollToFieldAndFocus(fieldId);\r\n      });\r\n    }\r\n  };\r\n  \r\n  var show_error_messages = function(message) {\r\n    \/\/ Show error message in the validation summary area\r\n    \/\/ Used when AJAX submission fails (network error, server error, etc.)\r\n    var summaryArea = thisform.find('.pfbc-validation-summary');\r\n    if (summaryArea.length) {\r\n      summaryArea.html('<div class=\"pfbc-validation-error\" role=\"alert\"><strong>' + accua_forms_i18n.check_fields + '<\/strong><ul>' + message + '<\/ul><\/div>').show();\r\n    }\r\n  }\r\n  \r\n  \/\/ Update summary area with server-side errors (called from AJAX error response)\r\n  var updateSummaryWithServerErrors = function(errors, elementErrors) {\r\n    \/\/ Collect server errors for the summary\r\n    fieldErrorsList = [];\r\n    \r\n    if (elementErrors) {\r\n      jQuery.each(elementErrors, function(fieldName, fieldErrors) {\r\n        var field = jQuery('[name=\"' + fieldName + '\"]', thisform);\r\n        var fieldId = field.attr('id') || fieldName;\r\n        var fieldLabel = getFieldLabel(field);\r\n        \r\n        \/\/ If no label found, try to get from the field container or use field name\r\n        if (!fieldLabel || fieldLabel === fieldName) {\r\n          var container = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n          fieldLabel = container.find('label').first().text().replace(\/\\s*\\*\\s*$\/, '').trim();\r\n          if (!fieldLabel) {\r\n            \/\/ Fallback: humanize the field name\r\n            fieldLabel = fieldName.replace(\/[-_]\/g, ' ').replace(\/\\b\\w\/g, function(l){ return l.toUpperCase(); });\r\n          }\r\n        }\r\n        \r\n        for (var i = 0; i < fieldErrors.length; i++) {\r\n          fieldErrorsList.push({\r\n            fieldId: fieldId,\r\n            label: fieldLabel,\r\n            errorType: fieldErrors[i]\r\n          });\r\n        }\r\n      });\r\n    }\r\n    \r\n    \/\/ Also add any general errors from the errors array\r\n    if (errors && errors.length > 0) {\r\n      for (var i = 0; i < errors.length; i++) {\r\n        \/\/ Check if this error is already in fieldErrorsList\r\n        var alreadyAdded = false;\r\n        for (var j = 0; j < fieldErrorsList.length; j++) {\r\n          if (fieldErrorsList[j].errorType === errors[i]) {\r\n            alreadyAdded = true;\r\n            break;\r\n          }\r\n        }\r\n        if (!alreadyAdded) {\r\n          fieldErrorsList.push({\r\n            fieldId: '',\r\n            label: '',\r\n            errorType: errors[i]\r\n          });\r\n        }\r\n      }\r\n    }\r\n    \r\n    \/\/ Update the summary area to show errors\r\n    if (fieldErrorsList.length > 0) {\r\n      updateSummaryArea(false);\r\n    }\r\n  }\r\n\r\n  _handle_ajax_submit_accua_form___accua_form__1_69ea045000f29 = function() {\r\n    if (_ajax_submitting_accua_form___accua_form__1_69ea045000f29) {\r\n      return false;\r\n    }\r\njQuery(\"#accua-form___accua-form__1_69ea045000f29 .pfbc-error\").remove();\r\n    \/\/ Mark that submit was attempted\r\n    submitAttempted = true;\r\n    \r\n    var valid_empty = true;\r\n    var valid_mail = true;\r\n    var valid_phone = true;\r\n    var fieldErrors = {};\r\n    \r\n    \/\/ Reset field errors list for summary\r\n    fieldErrorsList = [];\r\n\r\n    $(\"#accua-form___accua-form__1_69ea045000f29 .pfbc-element\").removeClass('pfbc-invalid');\r\n    $(\"#accua-form___accua-form__1_69ea045000f29 .pfbc-element\").removeClass('pfbc-element-has-error');\r\n    $(\"#accua-form___accua-form__1_69ea045000f29 input, #accua-form___accua-form__1_69ea045000f29 textarea, #accua-form___accua-form__1_69ea045000f29 select\").attr('aria-invalid', 'false');\r\n    $(\"#accua-form___accua-form__1_69ea045000f29 .pfbc-inline-error\").remove();\r\n\r\n    var processedGroups = {}; \/\/ Track radio\/checkbox groups to avoid duplicate errors\r\n\r\n    $('.accuaforms-field-required', thisform).each(function(){\r\n      var field = $(this);\r\n      var type = field.attr('type');\r\n      var fieldName = field.attr('name');\r\n\r\n      if (type === 'checkbox' || type === 'radio') {\r\n        \/\/ Skip if we've already processed this group\r\n        if (processedGroups[fieldName]) {\r\n          return true;\r\n        }\r\n        processedGroups[fieldName] = true;\r\n\r\n        if ($(\"[name='\"+fieldName+\"']:checked\", \"#accua-form___accua-form__1_69ea045000f29\").length > 0) {\r\n          return true;\r\n        }\r\n\r\n        valid_empty = false;\r\n        \r\n        \/\/ Find the container for the radio\/checkbox group\r\n        var groupContainer = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n        groupContainer.addClass('pfbc-invalid pfbc-element-has-error');\r\n        \r\n        \/\/ Remove any existing errors before adding new one (prevents duplicates)\r\n        removeGroupErrors(field, fieldName);\r\n        \r\n        \/\/ Get consistent error ID using helper\r\n        var errorId = getGroupErrorId(field);\r\n        \r\n        \/\/ Find the last radio\/checkbox in the group\r\n        var lastInGroup = $(\"[name='\"+fieldName+\"']\", \"#accua-form___accua-form__1_69ea045000f29\").last();\r\n        \r\n        \/\/ Apply ARIA attributes to all inputs in the group\r\n        $(\"[name='\"+fieldName+\"']\", \"#accua-form___accua-form__1_69ea045000f29\").attr('aria-invalid', 'true');\r\n        $(\"[name='\"+fieldName+\"']\", \"#accua-form___accua-form__1_69ea045000f29\").attr('aria-describedby', errorId);\r\n        \r\n        \/\/ Get field label and build error message\r\n        var fieldLabel = getFieldLabel(field);\r\n        var firstInput = $(\"[name='\"+fieldName+\"']\", \"#accua-form___accua-form__1_69ea045000f29\").first();\r\n        var errorMessage = getRequiredMessage(fieldLabel, firstInput);\r\n        \r\n        \/\/ Add to field errors list for summary (use first input ID for focusing)\r\n        fieldErrorsList.push({\r\n          fieldId: firstInput.attr('id'),\r\n          label: fieldLabel,\r\n          errorType: \"required field\"\r\n        });\r\n        \r\n        \/\/ Add error message after the last item in the group\r\n        var inlineError = $('<div class=\"pfbc-inline-error\" id=\"' + errorId + '\" role=\"alert\" aria-live=\"polite\"><div class=\"pfbc-error-message\">' + errorMessage + '<\/div><\/div>');\r\n        \r\n        \/\/ Insert after the last radio\/checkbox wrapper\r\n        var lastWrapper = lastInGroup.closest('.pfbc-radio, .pfbc-checkbox');\r\n        if (lastWrapper.length) {\r\n          lastWrapper.after(inlineError);\r\n        } else {\r\n          lastInGroup.after(inlineError);\r\n        }\r\n      } else {\r\n        var val = field.val();\r\n        if (typeof(val) == \"string\") {\r\n          \/\/ Treat \"-\" and \"Select...\" as invalid only for dropdowns\r\n          \/\/ For telephone fields, prefix-only values (e.g. \"+39\") are also empty\r\n          var isSelect = field.is('select');\r\n          if (! val.match(\/^\\s*$\/) && (!isSelect || (val !== \"Select...\" && val !== \"-\")) && !isTelephonePrefixOnly(field)) {\r\n            return true;\r\n          }\r\n        } else if (Array.isArray(val)) {\r\n          \/\/ Multiselect: empty array means no selection\r\n          if (val.length > 0) {\r\n            return true;\r\n          }\r\n        } else if (val) {\r\n          return true;\r\n        }\r\n\r\n        valid_empty = false;\r\n        var parent = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n        parent.addClass('pfbc-invalid pfbc-element-has-error');\r\n        \r\n        \/\/ Apply ARIA attributes\r\n        field.attr('aria-invalid', 'true');\r\n        var errorId = field.attr('id') + '-error';\r\n        field.attr('aria-describedby', errorId);\r\n        \r\n        \/\/ Get field label and build error message\r\n        var fieldLabel = getFieldLabel(field);\r\n        var errorMessage = getRequiredMessage(fieldLabel, field);\r\n        fieldErrorsList.push({\r\n          fieldId: field.attr('id'),\r\n          label: fieldLabel,\r\n          errorType: \"required field\"\r\n        });\r\n        \r\n        var inlineError = $('<div class=\"pfbc-inline-error\" id=\"' + errorId + '\" role=\"alert\" aria-live=\"polite\"><div class=\"pfbc-error-message\">' + errorMessage + '<\/div><\/div>');\r\n        \r\n        \/\/ For file inputs with help text, insert error after help text\r\n        var helpText = field.siblings('.pfbc-help').last();\r\n        if (field.is('[type=\"file\"]') && helpText.length) {\r\n          helpText.after(inlineError);\r\n        } else {\r\n          field.after(inlineError);\r\n        }\r\n      }\r\n    });\r\n\r\n    $('.pfbc-textbox[type=\"email\"]', thisform).each(function(){\r\n      var field = $(this);\r\n\r\n      if (field.val().match(\/^\\s*$\/)) {\r\n        return true;\r\n      }\r\n\r\n      if (field.val().match( \/^([a-zA-Z0-9_.+%-])+@(([a-zA-Z0-9-])+\\.)+([a-zA-Z0-9])+$\/ )) {\r\n        return true;\r\n      }\r\n\r\n      valid_mail = false;\r\n      var parent = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n      parent.addClass('pfbc-invalid pfbc-element-has-error');\r\n      \r\n      \/\/ Apply ARIA attributes\r\n      field.attr('aria-invalid', 'true');\r\n      var errorId = field.attr('id') + '-error';\r\n      field.attr('aria-describedby', errorId);\r\n      \r\n      \/\/ Get field label and build error message\r\n      var fieldLabel = getFieldLabel(field);\r\n      var errorMessage = getEmailMessage(fieldLabel, field);\r\n      \r\n      \/\/ Add to field errors list for summary (use field ID for focusing)\r\n      fieldErrorsList.push({\r\n        fieldId: field.attr('id'),\r\n        label: fieldLabel,\r\n        errorType: \"invalid email address\"\r\n      });\r\n      \r\n      var inlineError = $('<div class=\\\"pfbc-inline-error\\\" id=\\\"' + errorId + '\\\" role=\\\"alert\\\" aria-live=\\\"polite\\\"><div class=\\\"pfbc-error-message\\\">' + errorMessage + '<\/div><\/div>');\r\n      \r\n      \/\/ For file inputs with help text, insert error after help text\r\n      var helpText = field.siblings('.pfbc-help').last();\r\n      if (field.is('[type=\\\"file\\\"]') && helpText.length) {\r\n        helpText.after(inlineError);\r\n      } else {\r\n        field.after(inlineError);\r\n      }\r\n\r\n    });\r\n\r\n    \/\/ Phone validation - runs for all non-empty phone fields\r\n    $('.accuaform-telephone', thisform).each(function(){\r\n        var field = $(this);\r\n        var value = field.val();\r\n        \r\n        \/\/ Skip empty fields (Required validation handles mandatory)\r\n        if (!value || value.trim() === '') {\r\n          return true;\r\n        }\r\n        \r\n        \/\/ Skip prefix-only values (e.g. \"+39\") \u2014 treated as empty\r\n        var phoneTrimmed = value.trim();\r\n        if (phoneTrimmed.charAt(0) === '+' && phoneTrimmed.replace(\/\\D\/g, '').length <= 4) {\r\n          return true;\r\n        }\r\n        \r\n        \/\/ Get country code from data attribute\r\n        var countryCode = field.attr('data-country') || 'IT';\r\n        \r\n        \/\/ Use AccuaPhoneValidation if available, otherwise skip validation\r\n        if (typeof window.AccuaPhoneValidation !== 'undefined' && window.AccuaPhoneValidation.isValid) {\r\n          if (window.AccuaPhoneValidation.isValid(value, countryCode)) {\r\n            return true;\r\n          }\r\n        } else {\r\n          \/\/ Fallback: basic validation matching server-side Phone.php\r\n          var trimmed = value.trim();\r\n          \/\/ Check for invalid characters\r\n          if (!\/^[\\d\\s\\-\\.\\\/\\(\\)\\+]+$\/.test(trimmed)) {\r\n            \/\/ Invalid characters - fail validation\r\n          } else {\r\n            var plusIndex = trimmed.indexOf('+');\r\n            if (plusIndex > 0 || (trimmed.match(\/\\+\/g) || []).length > 1) {\r\n              \/\/ Plus in wrong position or multiple plus signs - fail validation\r\n            } else {\r\n              var digitCount = trimmed.replace(\/\\D\/g, '').length;\r\n              \/\/ Prefix-only (1-4 digits with +) treated as empty\r\n              if (trimmed.charAt(0) === '+' && digitCount <= 4) {\r\n                return true;\r\n              }\r\n              \/\/ Valid if 5-15 digits (matches server-side Phone.php)\r\n              if (digitCount >= 5 && digitCount <= 15) {\r\n                return true;\r\n              }\r\n            }\r\n          }\r\n        }\r\n        \r\n        valid_phone = false;\r\n        var parent = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n        parent.addClass('pfbc-invalid pfbc-element-has-error');\r\n        \r\n        \/\/ Remove any existing blur validation error to avoid duplicates\r\n        $('#' + field.attr('id') + '-phone-error').remove();\r\n        \r\n        \/\/ Apply ARIA attributes\r\n        field.attr('aria-invalid', 'true');\r\n        var errorId = field.attr('id') + '-error';\r\n        field.attr('aria-describedby', errorId);\r\n        \r\n        \/\/ Get field label and build error message\r\n        var fieldLabel = getFieldLabel(field);\r\n        var errorMessage = getPhoneMessage(fieldLabel, field);\r\n        \r\n        \/\/ Add to field errors list for summary\r\n        fieldErrorsList.push({\r\n          fieldId: field.attr('id'),\r\n          label: fieldLabel,\r\n          errorType: \"invalid phone number\"\r\n        });\r\n        \r\n        var inlineError = $('<div class=\"pfbc-inline-error pfbc-phone-format-error\" id=\"' + errorId + '\" role=\"alert\" aria-live=\"polite\"><div class=\"pfbc-error-message\">' + errorMessage + '<\/div><\/div>');\r\n        \r\n        var helpText = field.siblings('.pfbc-help').last();\r\n        if (helpText.length) {\r\n          helpText.after(inlineError);\r\n        } else {\r\n          field.after(inlineError);\r\n        }\r\n      });\r\n\r\n    if (valid_empty && valid_mail && valid_phone) {\r\n      \/\/ Show loading state in summary during AJAX submission\r\n      updateSummaryArea('loading');\r\n      \r\n      _ajax_submitting_accua_form___accua_form__1_69ea045000f29 = true;\r\n      disableSubmitButton();\r\n      $('input[name=\"_AccuaForm_tentatives\"]', thisform).val(fail_count);\r\n      disabled_fields = $(\"input, textarea, button, select\", thisform).not('[type=\"submit\"]').not(':disabled');\r\n      disabled_fields.attr('readonly','readonly');\r\n      timeout_count = 0;\r\n      if (ajax_enabled) {\r\n        $(\"#submit_target_accua_form___accua_form__1_69ea045000f29\").attr('src','').removeAttr('src');\r\n        timeout_handler = setTimeout(_handle_ajax_submit_timeout_accua_form___accua_form__1_69ea045000f29, 5000);\r\n      }\r\n      return true;\r\n    } else {\r\n      ga_submit_event('formSubmitInvalid');\r\n      \r\n      \/\/ Update URL hash to reflect invalid state (for GA tracking and bookmarkability)\r\n      if (history.replaceState) {\r\n        history.replaceState(null, '', '#formSubmitInvalid-'+anchor_id);\r\n      }\r\n      \r\n      \/\/ Update summary area with error list\r\n      updateSummaryArea(false);\r\n      \r\n      \/\/ Focus on first invalid field for accessibility\r\n      focusFirstInvalidField();\r\n      \r\n      return false;\r\n    }\r\n  }\r\n\r\n\r\n  _handle_ajax_submit_timeout_accua_form___accua_form__1_69ea045000f29 = function() {\r\n    if (_ajax_submitting_accua_form___accua_form__1_69ea045000f29) {\r\n      if (timeout_count < 60) {\r\n        timeout_count++;\r\n        timeout_handler = setTimeout(_handle_ajax_submit_timeout_accua_form___accua_form__1_69ea045000f29, 500);\r\n        _handle_ajax_submit_complete_accua_form___accua_form__1_69ea045000f29();\r\n      } else {\r\n        timeout_handler = false;\r\n        _handle_ajax_submit_complete_accua_form___accua_form__1_69ea045000f29();\r\n        if (_ajax_submitting_accua_form___accua_form__1_69ea045000f29) {\r\n          _handle_ajax_submit_response_accua_form___accua_form__1_69ea045000f29(false);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  _handle_ajax_submit_complete_accua_form___accua_form__1_69ea045000f29 = function() {\r\n    if (_ajax_submitting_accua_form___accua_form__1_69ea045000f29) {\r\n      var response = false;\r\n      try {\r\n        var responsedoc = frames['submit_target_accua_form___accua_form__1_69ea045000f29'].document;\r\n        if (responsedoc.getElementById(\"accua-form-ajax-response-loaded\")) {\r\n          response = $.parseJSON(responsedoc.getElementById(\"accua-form-ajax-response\").innerHTML);\r\n        }\r\n      } catch (err) {\r\n        response = false;\r\n      }\r\n      if (response) {\r\n        return _handle_ajax_submit_response_accua_form___accua_form__1_69ea045000f29 (response);\r\n      }\r\n    }\r\n  }\r\n\r\n  _handle_ajax_submit_message_accua_form___accua_form__1_69ea045000f29 = function(message) {\r\n    if (_ajax_submitting_accua_form___accua_form__1_69ea045000f29) {\r\n      var response = false;\r\n      try {\r\n        response = $.parseJSON(message.data);\r\n        \/\/ Accept response if jsuuid matches AND buildID matches or is null (server rejection)\r\n        if (response.jsuuid != jsuuid || (response.buildID != null && response.buildID != \"accua-form___accua-form__1_69ea045000f29\")) {\r\n          response = false;\r\n        }\r\n      } catch (err) {\r\n        response = false;\r\n      }\r\n      if (response) {\r\n        return _handle_ajax_submit_response_accua_form___accua_form__1_69ea045000f29 (response);\r\n      }\r\n    }\r\n  }\r\n\r\n  _handle_ajax_submit_response_accua_form___accua_form__1_69ea045000f29 = function(response) {\r\n    if (_ajax_submitting_accua_form___accua_form__1_69ea045000f29) {\r\n      if(response && typeof(response) == \"object\" && typeof(response.submitted) == \"boolean\") {\r\n        \/\/ Only show message container if there's actual content\r\n        if (response.messages && response.messages.trim() !== '') {\r\n          response_messages.html(response.messages).show();\r\n        } else {\r\n          response_messages.empty().hide();\r\n        }\r\n        if (response.submitted) {\r\n          if (response.valid) {\r\n            var gads_track_code = \"\";\r\n            if(gads_track_code != ''){            \r\n              gtag('event', 'conversion', {'send_to': gads_track_code});\r\n            }\r\n            \r\n            ga_submit_event('formSubmitSuccess');\r\n            smoothScrollToElement('formSubmitSuccess-'+anchor_id);\r\n            \r\n            $('#accua-form___accua-form__1_69ea045000f29').hide();          } else {\r\n            ga_submit_event('formSubmitInvalid');\r\n            smoothScrollToElement('formSubmitInvalid-'+anchor_id);var errorSize = response.errors.length;\r\nvar errorHTML = '';\r\nfor(e = 0; e < errorSize; ++e) {\r\n  errorHTML += '\\x3Cli\\x3E' + response.errors[e] + '\\x3C\/li\\x3E';\r\n}\r\n\r\n\/* Update summary area with server-side errors *\/\r\nif (typeof updateSummaryWithServerErrors === 'function') {\r\n  updateSummaryWithServerErrors(response.errors, response.elementErrors);\r\n}\r\n\r\n\/* Apply ARIA attributes and inline errors for accessibility *\/\r\nif (response.elementErrors) {\r\n  jQuery.each(response.elementErrors, function(fieldName, fieldErrors) {\r\n    var field = jQuery('[name=\"' + fieldName + '\"]');\r\n    if (field.length) {\r\n      var parent = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n      var errorId = field.attr('id') + '-error';\r\n      \r\n      \/* Add ARIA attributes *\/\r\n      field.attr('aria-invalid', 'true');\r\n      field.attr('aria-describedby', errorId);\r\n      \r\n      \/* Add error class to parent *\/\r\n      parent.addClass('pfbc-element-has-error');\r\n      \r\n      \/* Remove existing inline error if any *\/\r\n      parent.find('.pfbc-inline-error').remove();\r\n      \r\n      \/* Add inline error message *\/\r\n      var inlineErrorHTML = '<div class=\\\"pfbc-inline-error\\\" id=\\\"' + errorId + '\\\" role=\\\"alert\\\" aria-live=\\\"polite\\\">';\r\n      for (var i = 0; i < fieldErrors.length; i++) {\r\n        inlineErrorHTML += '<div class=\\\"pfbc-error-message\\\">' + fieldErrors[i] + '<\/div>';\r\n      }\r\n      inlineErrorHTML += '<\/div>';\r\n      \r\n      \/* For file inputs with help text, insert error after help text *\/\r\n      var helpText = field.siblings('.pfbc-help').last();\r\n      if (field.is('[type=\\\"file\\\"]') && helpText.length) {\r\n        helpText.after(inlineErrorHTML);\r\n      } else {\r\n        field.after(inlineErrorHTML);\r\n      }\r\n    }\r\n  });\r\n}\r\n            for (var name in response.files) {\r\n              $(\".pfbc-fieldwrap:has(input[type='file'][name='\"+name+\"'])\", thisform).html(response.files[name]);\r\n            }\r\n\r\n            $(\"input[name='_AccuaForm_hash']\",thisform).val(response._AccuaForm_hash);\r\n            $(\"input[name='_AccuaForm_iv']\",  thisform).val(response._AccuaForm_iv);\r\n            $(\"input[name='_AccuaForm_data']\",thisform).val(response._AccuaForm_data);\r\n\r\n            disabled_fields.removeAttr('readonly');\r\n            enableSubmitButton();\r\n          }\r\n        } else {\r\n          \/\/ Server did not recognize the form submission (e.g. expired nonce, stale cached page)\r\n          ga_submit_event('formSubmitError');\r\n          smoothScrollToElement('formSubmitError-'+anchor_id);\r\n          fail_count++;\r\n          if (fail_count > 2) {\r\n            ajax_enabled = false;\r\n            thisform.attr(\"action\", \"https:\\\/\\\/www.perustele.fi:443\\\/index.php\\\/wp-json\\\/wp\\\/v2\\\/pages\\\/2756\" );\r\n            thisform.removeAttr(\"target\");\r\n            $('input[name=\"_AccuaForm_submit_method\"]', thisform).val('fallback');\r\n          }\r\n          show_error_messages( \"\\u003Cli\\u003EForm submission failed. Please try again.\\u003C\\\/li\\u003E\" );\r\n          disabled_fields.removeAttr('readonly');\r\n          enableSubmitButton();\r\n        }\r\n      } else {\r\n        ga_submit_event('formSubmitError');\r\n        smoothScrollToElement('formSubmitError-'+anchor_id);\r\n        fail_count++;\r\n        if (fail_count > 2) {\r\n          ajax_enabled = false;\r\n          thisform.attr(\"action\", \"https:\\\/\\\/www.perustele.fi:443\\\/index.php\\\/wp-json\\\/wp\\\/v2\\\/pages\\\/2756\" );\r\n          thisform.removeAttr(\"target\");\r\n          $('input[name=\"_AccuaForm_submit_method\"]', thisform).val('fallback');\r\n        }\r\n        show_error_messages( \"\\u003Cli\\u003EForm submission failed. Please try again.\\u003C\\\/li\\u003E\" );\r\n        enableSubmitButton();\r\n      }\r\n      $('.accua_forms_show_recaptcha_button', thisform).click();\r\n      if (((typeof accuaform_recaptcha2_initialized) != 'undefined') && accuaform_recaptcha2_initialized) {\r\n        $('.accua_forms_recaptcha2_container', thisform).each(function(){\r\n          accua_forms_reload_recaptcha2($(this).attr('id'));\r\n        });\r\n      }\r\n      _ajax_submitting_accua_form___accua_form__1_69ea045000f29 = false;\r\n      if (timeout_handler) {\r\n        clearTimeout(timeout_handler);\r\n        timeout_handler = false;\r\n      }\r\n    }\r\n  }\r\n\r\n  if (ajax_enabled) {\r\n    thisform.attr(\"action\", \"https:\\\/\\\/www.perustele.fi\\\/wp-admin\\\/admin-ajax.php?action=accua_form_submit\" );\r\n    thisform.attr(\"target\",\"submit_target_accua_form___accua_form__1_69ea045000f29\");\r\n    try {\r\n      window.addEventListener('message', _handle_ajax_submit_message_accua_form___accua_form__1_69ea045000f29, false);\r\n    } catch (e) { }\r\n    $('input[name=\"_AccuaForm_submit_method\"]', thisform).val('iframe');\r\n  } else {\r\n    thisform.attr(\"action\", \"https:\\\/\\\/www.perustele.fi:443\\\/index.php\\\/wp-json\\\/wp\\\/v2\\\/pages\\\/2756\" );\r\n  }\r\n  thisform.attr(\"onsubmit\",\"return _handle_ajax_submit_accua_form___accua_form__1_69ea045000f29()\");\r\n  \r\n  \/\/ Real-time validation for better UX - use change only for checkbox\/radio to avoid double-firing\r\n  $('.accuaforms-field-required', thisform).on('change', function() {\r\n    var field = $(this);\r\n    var type = field.attr('type');\r\n    if (type !== 'checkbox' && type !== 'radio') return; \/\/ Only handle checkbox\/radio on change\r\n    \r\n    var fieldName = field.attr('name');\r\n    var isChecked = $(\"[name='\"+fieldName+\"']:checked\", thisform).length > 0;\r\n    var groupContainer = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n    var hasError = groupContainer.hasClass('pfbc-element-has-error');\r\n    \r\n    \/\/ Only act if state actually changed to avoid flashing\r\n    if (isChecked && hasError) {\r\n      \/\/ Valid now - remove error with animation\r\n      groupContainer.removeClass('pfbc-invalid pfbc-element-has-error');\r\n      $(\"[name='\"+fieldName+\"']\", thisform).attr('aria-invalid', 'false');\r\n      $(\"[name='\"+fieldName+\"']\", thisform).removeAttr('aria-describedby');\r\n      removeGroupErrors(field, fieldName, true); \/\/ animate=true\r\n    } else if (!isChecked && !hasError && submitAttempted) {\r\n      \/\/ Invalid now and we've attempted submit - show error\r\n      var errorId = getGroupErrorId(field);\r\n      groupContainer.addClass('pfbc-invalid pfbc-element-has-error');\r\n      $(\"[name='\"+fieldName+\"']\", thisform).attr('aria-invalid', 'true');\r\n      $(\"[name='\"+fieldName+\"']\", thisform).attr('aria-describedby', errorId);\r\n      \r\n      var fieldLabel = getFieldLabel(groupContainer);\r\n      var firstField = $(\"[name='\"+fieldName+\"']\", thisform).first();\r\n      var errorMessage = getRequiredMessage(fieldLabel, firstField);\r\n      var inlineError = $('<div class=\"pfbc-inline-error\" id=\"' + errorId + '\" role=\"alert\" aria-live=\"polite\"><div class=\"pfbc-error-message\">' + errorMessage + '<\/div><\/div>');\r\n      \r\n      var lastInGroup = $(\"[name='\"+fieldName+\"']\", thisform).last();\r\n      var lastWrapper = lastInGroup.closest('.pfbc-radio, .pfbc-checkbox');\r\n      if (lastWrapper.length) {\r\n        lastWrapper.after(inlineError);\r\n      } else {\r\n        lastInGroup.after(inlineError);\r\n      }\r\n    }\r\n  });\r\n  \r\n  \/\/ Blur handler for text-like fields only\r\n  $('.accuaforms-field-required', thisform).on('blur', function() {\r\n    var field = $(this);\r\n    var type = field.attr('type');\r\n    var fieldName = field.attr('name');\r\n    var isEmpty = false;\r\n    \r\n    \/\/ Skip checkbox\/radio - handled by change event above\r\n    if (type === 'checkbox' || type === 'radio') return;\r\n    \r\n    var val = field.val();\r\n      if (typeof(val) == \"string\") {\r\n        \/\/ Treat \"-\" and \"Select...\" as invalid only for dropdowns\r\n        \/\/ For telephone fields, prefix-only values (e.g. \"+39\") are also empty\r\n        var isSelect = field.is('select');\r\n        isEmpty = val.match(\/^\\s*$\/) || (isSelect && (val === \"Select...\" || val === \"-\")) || isTelephonePrefixOnly(field);\r\n      } else if (typeof(val) == \"object\") {\r\n        isEmpty = !val || val.length === 0;\r\n      } else {\r\n        isEmpty = !val;\r\n      }\r\n      \r\n      var parent = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n      var errorId = field.attr('id') + '-error';\r\n      \r\n      if (isEmpty) {\r\n        \/\/ For telephone: phone-validation.js blur handler runs AFTER this one and may\r\n        \/\/ still have pfbc-element-has-error set from a previous format error. Check for\r\n        \/\/ the specific required error div instead of the parent class to avoid skipping.\r\n        var alreadyHasError = field.hasClass('accuaform-telephone')\r\n          ? $('#' + errorId).length > 0\r\n          : parent.hasClass('pfbc-element-has-error');\r\n        if (!alreadyHasError) {\r\n          \/\/ For telephone: remove leftover phone format error since the field is now\r\n          \/\/ empty (required error takes priority). phone-validation.js will also clean\r\n          \/\/ up on its blur, but this handler runs first.\r\n          if (field.hasClass('accuaform-telephone')) {\r\n            $('#' + field.attr('id') + '-phone-error').remove();\r\n          }\r\n          parent.addClass('pfbc-invalid pfbc-element-has-error');\r\n          field.attr('aria-invalid', 'true');\r\n          field.attr('aria-describedby', errorId);\r\n          \r\n          \/\/ Get field label for error message\r\n          var fieldLabel = getFieldLabel(parent);\r\n          var errorMessage = getRequiredMessage(fieldLabel, field);\r\n          var inlineError = $('<div class=\"pfbc-inline-error\" id=\"' + errorId + '\" role=\"alert\" aria-live=\"polite\"><div class=\"pfbc-error-message\">' + errorMessage + '<\/div><\/div>');\r\n          \r\n          \/\/ For file inputs with help text, insert error after help text\r\n          var helpText = field.siblings('.pfbc-help').last();\r\n          if (field.is('[type=\"file\"]') && helpText.length) {\r\n            helpText.after(inlineError);\r\n          } else {\r\n            field.after(inlineError);\r\n          }\r\n        }\r\n      } else {\r\n        \/\/ Field is not empty \u2014 clear required-related errors.\r\n        if (field.hasClass('accuaform-telephone')) {\r\n          \/\/ For telephone fields: clear only the required error ({id}-error).\r\n          \/\/ phone-validation.js has already run on this same blur event and set the\r\n          \/\/ correct error state (phone-error or clean). We must not undo its work.\r\n          \/\/ Only remove the required-error div; preserve phone-validation.js state.\r\n          $('#' + errorId).remove();\r\n          \/\/ If phone-validation.js left no errors, clear the container state too\r\n          if (!parent.find('.pfbc-inline-error').length) {\r\n            parent.removeClass('pfbc-invalid pfbc-element-has-error');\r\n            field.attr('aria-invalid', 'false');\r\n            field.removeAttr('aria-describedby');\r\n          }\r\n        } else {\r\n          \/\/ For non-telephone fields: animated removal for smoother UX\r\n          removeErrorAnimated(errorId);\r\n          parent.removeClass('pfbc-invalid pfbc-element-has-error');\r\n          field.attr('aria-invalid', 'false');\r\n          field.removeAttr('aria-describedby');\r\n        }\r\n      }\r\n  });\r\n  \r\n  $('.pfbc-textbox[type=\"email\"]', thisform).on('blur change', function() {\r\n    var field = $(this);\r\n    var val = field.val();\r\n    \r\n    if (val.match(\/^\\s*$\/)) {\r\n      return; \/\/ Empty is handled by required validation\r\n    }\r\n    \r\n    var isValid = val.match(\/^([a-zA-Z0-9_.+%-])+@(([a-zA-Z0-9-])+\\.)+([a-zA-Z0-9])+$\/);\r\n    var parent = field.closest('.pfbc-element, .pfbc-fieldwrap');\r\n    var errorId = field.attr('id') + '-error';\r\n    \r\n    if (!isValid) {\r\n      if (!parent.hasClass('pfbc-element-has-error')) {\r\n        parent.addClass('pfbc-invalid pfbc-element-has-error');\r\n        field.attr('aria-invalid', 'true');\r\n        field.attr('aria-describedby', errorId);\r\n        \r\n        \/\/ Get field label for error message\r\n        var fieldLabel = getFieldLabel(parent);\r\n        var errorMessage = getEmailMessage(fieldLabel, field);\r\n        var inlineError = $('<div class=\\\"pfbc-inline-error\\\" id=\\\"' + errorId + '\\\" role=\\\"alert\\\" aria-live=\\\"polite\\\"><div class=\\\"pfbc-error-message\\\">' + errorMessage + '<\/div><\/div>');\r\n        \r\n        \/\/ For file inputs with help text, insert error after help text\r\n        var helpText = field.siblings('.pfbc-help').last();\r\n        if (field.is('[type=\\\"file\\\"]') && helpText.length) {\r\n          helpText.after(inlineError);\r\n        } else {\r\n          field.after(inlineError);\r\n        }\r\n      }\r\n    } else {\r\n      parent.removeClass('pfbc-invalid pfbc-element-has-error');\r\n      field.attr('aria-invalid', 'false');\r\n      field.removeAttr('aria-describedby');\r\n      removeErrorAnimated(errorId);\r\n    }\r\n  });\r\n});\r\n\/\/ -->\r\n<\/script>\r\n<iframe id=\"submit_target_accua_form___accua_form__1_69ea045000f29\" title=\"Notification Message\" name=\"submit_target_accua_form___accua_form__1_69ea045000f29\" onload=\"_handle_ajax_submit_complete_accua_form___accua_form__1_69ea045000f29()\" onerror=\"_handle_ajax_submit_complete_accua_form___accua_form__1_69ea045000f29()\" style=\"width:0;height:0;border:0px solid #fff\"><\/iframe><script type=\"text\/javascript\">\r\n<!--\r\njQuery(function($){\r\n  var referrerfield = $(\"#accua-form___accua-form__1_69ea045000f29 input[name='_AccuaForm_referrer']\");\r\n  if (referrerfield.val() == '') {\r\n    referrerfield.val(document.referrer);\r\n  }\r\n  $(\"#accua-form___accua-form__1_69ea045000f29 input[name='_AccuaForm_user_agent']\").val(navigator.userAgent);\r\n  $(\"#accua-form___accua-form__1_69ea045000f29 input[name='_AccuaForm_platform']\").val(navigator.platform);\r\n});\r\n\/\/ -->\r\n<\/script>\n","protected":false},"excerpt":{"rendered":"<p>GXP1630 on varustettu joukolla helppok\u00e4ytt\u00f6isi\u00e4 VoIP-ominaisuuksia. GXP1630 on tarkoitettu ensisijaisesti pieniin ja keskisuuriin puhelum\u00e4\u00e4riin ja tehokkaaseen puhelujen k\u00e4sittelyyn. Siin\u00e4 on \u00a03 linjaa\/SIP-tili\u00e4 ja 8 kaksiv\u00e4rist\u00e4 BLF\/pikavalintan\u00e4pp\u00e4int\u00e4. GXP1630 tukee my\u00f6s parhaita mahdollisia yhteysnopeuksia ja puhelun laatua kahdella Gigabit-portilla ja HD-\u00e4\u00e4nell\u00e4 sek\u00e4 kaiutinpuhelimessa ett\u00e4 luurissa. Muiden ominaisuuksiensa, kuten integroidun PoE:n, 3 ohjelmoitavan XML-ohjelmiston\u00e4pp\u00e4imen ja 4-suuntaisen neuvottelutuen, ansiosta [&hellip;]<\/p>\n","protected":false},"author":1,"featured_media":0,"parent":0,"menu_order":0,"comment_status":"closed","ping_status":"closed","template":"","meta":{"footnotes":""},"class_list":["post-2756","page","type-page","status-publish","hentry"],"_links":{"self":[{"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/pages\/2756","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/pages"}],"about":[{"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/types\/page"}],"author":[{"embeddable":true,"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/comments?post=2756"}],"version-history":[{"count":4,"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/pages\/2756\/revisions"}],"predecessor-version":[{"id":2929,"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/pages\/2756\/revisions\/2929"}],"wp:attachment":[{"href":"https:\/\/www.perustele.fi\/index.php\/wp-json\/wp\/v2\/media?parent=2756"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}