diff --git a/none b/none new file mode 100644 index 000000000..44acdc4be --- /dev/null +++ b/none @@ -0,0 +1,9 @@ +{ + "version": 3, + "file": "assets/css/admin.css", + "sources": [ + "assets/css/admin.scss" + ], + "names": [], + "mappings": "AAAA,AAAA,KAAK,AAAA,WAAW,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,AAAA,WAAW,CAAC;EAClC,MAAM,EAAE,YAAY;EACpB,OAAO,EAAE,YAAY,GACtB;;AAED,AAAA,wBAAwB,CAAC;EACvB,QAAQ,EAAE,QAAQ;EAClB,QAAQ,EAAE,MAAM,GA8BjB;EAhCD,AAIE,wBAJsB,CAItB,EAAE,CAAC;IACD,eAAe,EAAE,IAAI;IACrB,YAAY,EAAE,KAAK,GACpB;EAPH,AASE,wBATsB,CAStB,eAAe,CAAC;IACd,QAAQ,EAAE,MAAM;IAChB,KAAK,EAAE,KAAK;IACZ,OAAO,EAAE,gBAAgB;IACzB,UAAU,EAAE,KAAK;IACjB,SAAS,EAAE,IAAI;IACf,WAAW,EAAE,KAAK;IAClB,eAAe,EAAE,IAAI,GAStB;IAzBH,AAkBI,wBAlBoB,CAStB,eAAe,AASZ,QAAQ,CAAC;MACR,QAAQ,EAAE,QAAQ;MAClB,GAAG,EAAE,IAAI;MACT,IAAI,EAAE,KAAK;MACX,kBAAkB,EAAE,mBAAmB;MACvC,UAAU,EAAE,mBAAmB,GAChC;EAxBL,AA4BI,wBA5BoB,CA2BtB,CAAC,AAAA,OAAO,CACN,OAAO,CAAC;IACN,YAAY,EAAE,GAAG,GAClB;;AAIL,AAAA,wBAAwB,CAAC,gBAAgB,CAAC;EACxC,UAAU,EAAE,IAAI,GACjB;;AAED,AAAA,4BAA4B,CAAC,gBAAgB,CAAC;EAC5C,UAAU,EAAE,IAAI,GACjB;;AAED,AAAA,8BAA8B,CAAC;EAC7B,UAAU,EAAE,cAAc,GAC3B;;AAED,AAAA,kCAAkC,CAAC;EACjC,MAAM,EAAE,gBAAgB;EACxB,UAAU,EAAE,UAAU;EACtB,OAAO,EAAE,IAAI;EACb,OAAO,EAAE,QAAQ,GAClB;;AAED,AAAA,qBAAqB,EAAE,yBAAyB,EAAE,sBAAsB,EAAE,0BAA0B,CAAE;EACpG,KAAK,EAAE,eAAe,GACvB;;AAED,AAAA,cAAc,CAAC;EACb,gBAAgB,EAAE,kBAAkB;EACpC,gBAAgB,EAAE,gDAAgD,CAAC,UAAU;EAC7E,KAAK,EAAE,eAAe;EACtB,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,sBAAsB,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAE,KAAI,CAAC,oBAAoB,CAAC,UAAU;EACpG,MAAM,EAAE,gCAAgC;EACxC,WAAW,EAAE,eAAe;EAC5B,UAAU,EAAE,uFAAuF,GA4BpG;EAnCD,AASE,cATY,AASX,MAAM,CAAC;IACN,iBAAiB,EAAE,WAAW;IAC9B,SAAS,EAAE,WAAW;IACtB,MAAM,EAAE,OAAO,GAChB;EAbH,AAeE,cAfY,CAeZ,WAAW,CAAC;IACV,MAAM,EAAE,KAAK;IACb,UAAU,EAAE,IAAI;IAChB,KAAK,EAAE,IAAI,GAKZ;IAvBH,AAoBI,cApBU,CAeZ,WAAW,AAKR,MAAM,CAAC;MACN,KAAK,EAAE,IAAI,GACZ;EAtBL,AAyBE,cAzBY,CAyBZ,oBAAoB,CAAC;IACnB,MAAM,EAAE,KAAK;IACb,KAAK,EAAE,IAAI;IACX,YAAY,EAAE,IAAI;IAClB,UAAU,EAAE,WAAW,GAKxB;IAlCH,AA+BI,cA/BU,CAyBZ,oBAAoB,AAMjB,MAAM,CAAC;MACN,KAAK,EAAE,IAAI,GACZ;;AAIL,AAAA,sBAAsB,CAAC,OAAO,CAAC;EAC7B,YAAY,EAAE,GAAG,GAClB;;AAED,AAAA,WAAW,EAAE,oBAAoB,CAAC,CAAC,AAAA,OAAO,EAAE,qBAAqB,AAAA,OAAO,CAAC;EACvE,SAAS,EAAE,GAAG;EACd,aAAa,EAAE,GAAG;EAClB,WAAW,EAAE,GAAG;EAChB,KAAK,EAAE,IAAI;EACX,UAAU,EAAE,MAAM;EAClB,cAAc,EAAE,SAAS;EACzB,OAAO,EAAE,OAAO;EAChB,QAAQ,EAAE,QAAQ;EAClB,GAAG,EAAE,IAAI;EACT,UAAU,EAAE,IAAI;EAChB,UAAU,EAAE,MAAM;EAClB,OAAO,EAAE,YAAY;EACrB,eAAe,EAAE,IAAI,GACtB;;AAED,AACE,CADD,AAAA,WAAW,AAAA,IAAK,CAAA,oBAAoB,CAClC,MAAM,CAAC;EACN,UAAU,EAAE,OAAO;EACnB,KAAK,EAAE,IAAI,GACZ;;AAGH,AAAA,oBAAoB,EAAE,oBAAoB,CAAC,CAAC,AAAA,OAAO,EAAE,qBAAqB,AAAA,OAAO,CAAC;EAChF,WAAW,EAAE,GAAG;EAChB,UAAU,EAAE,GAAG;EACf,MAAM,EAAE,cAAc;EACtB,KAAK,EAAE,IAAI;EACX,UAAU,EAAE,WAAW,GAKxB;EAVD,AAOE,oBAPkB,AAOjB,MAAM,EAPa,oBAAoB,CAAC,CAAC,AAAA,OAAO,AAOhD,MAAM,EAP4C,qBAAqB,AAAA,OAAO,AAO9E,MAAM,CAAC;IACN,KAAK,EAAE,IAAI,GACZ;;AAGH,AAAA,qBAAqB,AAAA,OAAO,CAAC;EAC3B,OAAO,EAAE,KAAK;EACd,WAAW,EAAE,MAAM,GACpB;;AAED,AAEI,oBAFgB,CAClB,CAAC,AACE,OAAO,CAAC;EACP,WAAW,EAAE,CAAC;EACd,KAAK,EAAE,OAAO;EACd,YAAY,EAAE,OAAO;EACrB,OAAO,EAAE,KAAK,GACf;;AAPL,AASI,oBATgB,CAClB,CAAC,AAQE,MAAM,AAAA,OAAO,CAAC;EACb,YAAY,EAAE,OAAO;EACrB,KAAK,EAAE,OAAO,GACf;;AAZL,AAeE,oBAfkB,AAejB,OAAO,CAAC,CAAC,AAAA,OAAO,CAAC;EAChB,YAAY,EAAE,IAAI;EAClB,KAAK,EAAE,IAAI,GACZ;;AAGH,AAAA,uBAAuB,CAAC;EACtB,iBAAiB,EAAE,kBAAkB,GACtC;;AAED,AACE,CADD,CACC,oBAAoB,CAAC;EACnB,eAAe,EAAE,IAAI;EACrB,MAAM,EAAE,iBAAiB;EACzB,KAAK,EAAE,OAAO,GAKf;EATH,AAMI,CANH,CACC,oBAAoB,AAKjB,MAAM,CAAC;IACN,KAAK,EAAE,OAAO,GACf;;AAIL,AAAA,CAAC,AAAA,oBAAoB,CAAC;EACpB,eAAe,EAAE,IAAI;EACrB,MAAM,EAAE,iBAAiB;EACzB,WAAW,EAAE,IAAI;EACjB,KAAK,EAAE,kBAAkB;EACzB,MAAM,EAAE,KAAK,GAKd;EAVD,AAOE,CAPD,AAAA,oBAAoB,AAOlB,MAAM,CAAC;IACN,KAAK,EAAE,kBAAkB,GAC1B;;AAGH,AACE,CADD,AAAA,kBAAkB,CACjB,WAAW,CAAC;EACV,UAAU,EAAE,OAAO,GACpB;;AAHH,AAMI,CANH,AAAA,kBAAkB,AAKhB,MAAM,CACL,WAAW,AAAA,IAAK,CAAA,oBAAoB,EAAE;EACpC,UAAU,EAAE,OAAO,GACpB;;AAIL,AAAA,2BAA2B,CAAC;EAC1B,UAAU,EAAE,OAAO;EACnB,MAAM,EAAE,cAAc;EACtB,KAAK,EAAE,IAAI;EACX,OAAO,EAAE,KAAK,GACf;;AAED,AAAA,kBAAkB,CAAC,WAAW,CAAC;EAC7B,GAAG,EAAE,IAAI,GACV;;AAED,AACE,CADD,AAAA,gCAAgC,CAC/B,KAAK,AAAA,SAAS,CAAC;EACb,KAAK,EAAE,eAAe;EACtB,YAAY,EAAE,cAAc,GAC7B;;AAGH,AACE,CADD,AAAA,uBAAuB,CACtB,KAAK,CAAC;EACJ,OAAO,EAAE,gBAAgB;EACzB,aAAa,EAAE,GAAG,GACnB;;AAJH,AAME,CAND,AAAA,uBAAuB,CAMtB,oBAAoB,CAAC;EACnB,YAAY,EAAE,GAAG,GAClB;;AAGH,AAAA,uBAAuB,CAAC,KAAK,EAAE,uBAAuB,CAAC,KAAK,EAAE,uBAAuB,CAAC,oBAAoB,CAAC;EACzG,OAAO,EAAE,GAAG,GACb;;AAED,AAAA,KAAK,AAAA,mBAAmB,CAAC,EAAE,CAAC,EAAE,EAAE,KAAK,AAAA,mBAAmB,CAAC,EAAE,CAAC,EAAE,CAAC;EAC7D,OAAO,EAAE,GAAG;EACZ,SAAS,EAAE,KAAK,GACjB;;AAED,AAAA,KAAK,AAAA,mBAAmB,CAAC,EAAE,CAAC,EAAE,AAAA,YAAY,EAAE,KAAK,AAAA,mBAAmB,CAAC,EAAE,CAAC,EAAE,AAAA,YAAY,CAAC;EACrF,YAAY,EAAE,CAAC,GAChB;;AAED,AAAA,eAAe,CAAC;EACd,WAAW,EAAE,GAAG;EAChB,OAAO,EAAE,KAAK;EACd,KAAK,EAAE,IAAI;EACX,KAAK,EAAE,IAAI,GACZ;;AAED,AAAA,qBAAqB,AAAA,MAAM,CAAC;EAC1B,OAAO,EAAE,OAAO,GACjB;;AAED,AAAA,WAAW,CAAC,kBAAkB,CAAC,CAAC,AAAA,kBAAkB,CAAC;EACjD,KAAK,EAAE,IAAI;EACX,MAAM,EAAE,CAAC;EACT,OAAO,EAAE,QAAQ;EACjB,MAAM,EAAE,SAAS;EACjB,QAAQ,EAAE,MAAM;EAChB,QAAQ,EAAE,QAAQ;EAClB,KAAK,EAAE,IAAI;EACX,MAAM,EAAE,CAAC;EACT,KAAK,EAAE,KAAK,GACb;;AAED,AAAA,WAAW,CAAC,kBAAkB,CAAC,CAAC,AAAA,kBAAkB,AAAA,OAAO,CAAC;EACxD,WAAW,EAAE,SAAS;EACtB,OAAO,EAAE,OAAO;EAChB,QAAQ,EAAE,QAAQ;EAClB,GAAG,EAAE,CAAC;EACN,IAAI,EAAE,CAAC;EACP,UAAU,EAAE,MAAM;EAClB,cAAc,EAAE,GAAG;EACnB,WAAW,EAAE,IAAI;EACjB,SAAS,EAAE,IAAI;EACf,WAAW,EAAE,GAAG;EAChB,sBAAsB,EAAE,WAAW,GACpC;;AAED,AAAA,WAAW,CAAC,kBAAkB,CAAC,2BAA2B,EAAE,WAAW,CAAC,kBAAkB,CAAC,yBAAyB,EAAE,WAAW,CAAC,kBAAkB,CAAC,wBAAwB,EAAE,WAAW,CAAC,kBAAkB,CAAC,+BAA+B,CAAC;EAC5O,KAAK,EAAE,IAAI;EACX,KAAK,EAAE,eAAe,GACvB;;AAED,AAAA,WAAW,CAAC,kBAAkB,CAAC,2BAA2B,CAAC,KAAK,EAAE,WAAW,CAAC,kBAAkB,CAAC,yBAAyB,CAAC,KAAK,EAAE,WAAW,CAAC,kBAAkB,CAAC,wBAAwB,CAAC,KAAK,EAAE,WAAW,CAAC,kBAAkB,CAAC,+BAA+B,CAAC,KAAK,CAAC;EACpQ,KAAK,EAAE,IAAI,GACZ;;AAED,AAAA,WAAW,CAAC,uCAAuC,CAAC;EAClD,KAAK,EAAE,IAAI;EACX,KAAK,EAAE,eAAe,GACvB;;AAED,AAAA,gBAAgB,CAAC;EACf,KAAK,EAAE,GAAG,GACX;;AAED,AAAA,mBAAmB,CAAC;EAClB,WAAW,EAAE,GAAG,GACjB;;AAED,AAAA,kBAAkB,CAAC;EACjB,KAAK,EAAE,KAAK,GACb;;AAED,AAAA,cAAc,CAAC,IAAI,CAAC;EAClB,OAAO,EAAE,KAAK;EACd,WAAW,EAAE,OAAO;EACpB,QAAQ,EAAE,QAAQ;EAClB,OAAO,EAAE,CAAC,CAAA,UAAU;EACpB,MAAM,EAAE,GAAG,CAAA,UAAU;EACrB,KAAK,EAAE,GAAG,GACX;;AAED,AAAA,cAAc,CAAC,IAAI,AAAA,MAAM,CAAC;EACxB,OAAO,EAAE,OAAO;EAChB,WAAW,EAAE,WAAW;EACxB,WAAW,EAAE,CAAC;EACd,QAAQ,EAAE,QAAQ;EAClB,KAAK,EAAE,IAAI;EACX,MAAM,EAAE,IAAI;EACZ,KAAK,EAAE,IAAI;EACX,YAAY,EAAE,MAAM;EACpB,cAAc,EAAE,IAAI;EACpB,sBAAsB,EAAE,WAAW;EACnC,GAAG,EAAE,CAAC;EACN,IAAI,EAAE,CAAC;EACP,WAAW,EAAE,IAAI;EACjB,MAAM,EAAE,CAAC;EACT,UAAU,EAAE,MAAM;EAClB,WAAW,EAAE,GAAG,GACjB;;AAED,AAAA,gCAAgC,CAAC;EAC/B,KAAK,EAAE,IAAI;EACX,KAAK,EAAE,IAAI;EACX,KAAK,EAAE,IAAI,GACZ;;AAED,AAGM,sBAHgB,CACpB,uBAAuB,CACrB,kBAAkB,CAChB,0BAA0B,EAHhC,sBAAsB,CACK,qBAAqB,CAC5C,kBAAkB,CAChB,0BAA0B,CAAC;EACzB,MAAM,EAAE,IAAI,GAcb;EAlBP,AAMQ,sBANc,CACpB,uBAAuB,CACrB,kBAAkB,CAChB,0BAA0B,CAGxB,yBAAyB,EANjC,sBAAsB,CACK,qBAAqB,CAC5C,kBAAkB,CAChB,0BAA0B,CAGxB,yBAAyB,CAAC;IACxB,YAAY,EAAE,IAAI,GACnB;EART,AAUQ,sBAVc,CACpB,uBAAuB,CACrB,kBAAkB,CAChB,0BAA0B,CAOxB,4BAA4B,EAVpC,sBAAsB,CACK,qBAAqB,CAC5C,kBAAkB,CAChB,0BAA0B,CAOxB,4BAA4B,CAAC;IAC3B,WAAW,EAAE,IAAI,GAClB;EAZT,AAcQ,sBAdc,CACpB,uBAAuB,CACrB,kBAAkB,CAChB,0BAA0B,CAWxB,yBAAyB,EAdjC,sBAAsB,CACK,qBAAqB,CAC5C,kBAAkB,CAChB,0BAA0B,CAWxB,yBAAyB,CAAC;IACxB,KAAK,EAAE,GAAG;IACV,MAAM,EAAE,IAAI,GACb;;AAMT,AAAA,mDAAmD,CAAC;EAClD,OAAO,EAAE,IAAI,GACd;;AAED,AACE,sBADoB,CACpB,0CAA0C,CAAC;EACzC,UAAU,EAAE,IAAI;EAChB,aAAa,EAAE,IAAI,GACpB;;AAGH,AACE,gCAD8B,CAC9B,0CAA0C,CAAC;EACzC,aAAa,EAAE,CAAC,GACjB;;AAGH,AACE,qBADmB,CAAC,WAAW,AAAA,sBAAsB,CACrD,qCAAqC,CAAC;EACpC,MAAM,EAAE,KAAK,GAcd;EAhBH,AAII,qBAJiB,CAAC,WAAW,AAAA,sBAAsB,CACrD,qCAAqC,CAGnC,uBAAuB,AAAA,OAAO,CAAC;IAC7B,MAAM,EAAE,YAAY,GAUrB;IAfL,AAOM,qBAPe,CAAC,WAAW,AAAA,sBAAsB,CACrD,qCAAqC,CAGnC,uBAAuB,AAAA,OAAO,CAG5B,CAAC,CAAC;MACA,OAAO,EAAE,GAAG;MACZ,SAAS,EAAE,IAAI,GAChB;IAVP,AAYM,qBAZe,CAAC,WAAW,AAAA,sBAAsB,CACrD,qCAAqC,CAGnC,uBAAuB,AAAA,OAAO,CAQ5B,EAAE,CAAC;MACD,YAAY,EAAE,YAAY,GAC3B;;AAdP,AAkBE,qBAlBmB,CAAC,WAAW,AAAA,sBAAsB,CAkBrD,CAAC,AAAA,iCAAiC,CAAC;EACjC,KAAK,EAAE,IAAI;EACX,KAAK,EAAE,IAAI;EACX,KAAK,EAAE,IAAI;EACX,WAAW,EAAE,IAAI;EACjB,OAAO,EAAE,QAAQ;EACjB,WAAW,EAAE,IAAI;EACjB,SAAS,EAAE,GAAG;EACd,UAAU,EAAE,IAAI,GAKjB;EA/BH,AA4BI,qBA5BiB,CAAC,WAAW,AAAA,sBAAsB,CAkBrD,CAAC,AAAA,iCAAiC,CAUhC,kBAAkB,CAAC;IACjB,GAAG,EAAE,IAAI,GACV;;AAIL,AACE,0BADwB,CACxB,qCAAqC,CAAC;EACpC,OAAO,EAAE,IAAI,GAQd;EAVH,AAKM,0BALoB,CACxB,qCAAqC,CAGnC,uBAAuB,CACrB,CAAC,CAAC;IACA,OAAO,EAAE,GAAG;IACZ,SAAS,EAAE,IAAI,GAChB;;AARP,AAYE,0BAZwB,CAYxB,CAAC,AAAA,iCAAiC,CAAC;EACjC,OAAO,EAAE,mBAAmB;EAC5B,WAAW,EAAE,IAAI;EACjB,SAAS,EAAE,GAAG;EACd,UAAU,EAAE,IAAI;EAChB,UAAU,EAAE,IAAI,GAKjB;EAtBH,AAmBI,0BAnBsB,CAYxB,CAAC,AAAA,iCAAiC,CAOhC,kBAAkB,CAAC;IACjB,GAAG,EAAE,IAAI,GACV;;AArBL,AAwBE,0BAxBwB,CAwBxB,wBAAwB,CAAC;EACvB,KAAK,EAAE,GAAG,GAYX;EArCH,AA4BM,0BA5BoB,CAwBxB,wBAAwB,CAGtB,KAAK,CACH,KAAK,CAAC;IACJ,KAAK,EAAE,MAAM,GACd;EA9BP,AA+BM,0BA/BoB,CAwBxB,wBAAwB,CAGtB,KAAK,CAIH,KAAK,CAAC;IACJ,KAAK,EAAE,IAAI;IACX,YAAY,EAAE,IAAI;IAClB,WAAW,EAAE,EAAE,GAChB;;AAKP,AACE,6BAD2B,CAC3B,KAAK,CAAC;EACJ,OAAO,EAAE,KAAK;EACd,KAAK,EAAE,IAAI;EACX,aAAa,EAAE,GAAG,GACnB;;AAGH,AACE,+BAD6B,CAC7B,uBAAuB,CAAC;EACtB,YAAY,EAAE,IAAI,GACnB;;AAHH,AAKE,+BAL6B,CAK7B,uBAAuB,CAAC;EACtB,KAAK,EAAE,OAAO,GAKf;EAXH,AAQI,+BAR2B,CAK7B,uBAAuB,AAGpB,aAAa,CAAC;IACb,OAAO,EAAE,IAAI,GACd;;AAIL,AACE,wBADsB,CACtB,kBAAkB,CAAC,4BAA4B,CAAC;EAC9C,WAAW,EAAE,OAAO,GACrB;;AAHH,AAKE,wBALsB,CAKtB,2BAA2B,CAAC,4BAA4B,CAAC,EAAE,AAAA,uBAAuB,CAAC;EACjF,WAAW,EAAE,CAAC;EACd,UAAU,EAAE,CAAC,GAKd;EAZH,AASI,wBAToB,CAKtB,2BAA2B,CAAC,4BAA4B,CAAC,EAAE,AAAA,uBAAuB,CAIhF,KAAK,AAAA,sBAAsB,CAAC;IAC1B,UAAU,EAAE,IAAI,GACjB;;AAIL,AACE,4CAD0C,CAC1C,6CAA6C,CAAC;EAC5C,WAAW,EAAE,CAAC;EACd,QAAQ,EAAE,MAAM;EAChB,KAAK,EAAE,OAAO;EACd,eAAe,EAAE,IAAI;EACrB,cAAc,EAAE,MAAM,GACvB;;AAPH,AASE,4CAT0C,AASzC,sDAAsD,CAAC;EACtD,OAAO,EAAE,IAAI;EACb,SAAS,EAAE,IAAI;EACf,WAAW,EAAE,MAAM,GAcpB;EA1BH,AAcI,4CAdwC,AASzC,sDAAsD,CAKrD,KAAK,CAAC;IACJ,KAAK,EAAE,IAAI,GACZ;EAhBL,AAkBI,4CAlBwC,AASzC,sDAAsD,CASrD,kBAAkB,CAAC;IACjB,SAAS,EAAE,cAAc;IACzB,KAAK,EAAE,cAAc,GACtB;EArBL,AAuBI,4CAvBwC,AASzC,sDAAsD,CAcrD,6CAA6C,CAAC;IAC5C,WAAW,EAAE,IAAI,GAClB;;AAzBL,AA4BE,4CA5B0C,AA4BzC,0DAA0D,CAAC;EAC1D,OAAO,EAAE,IAAI;EACb,WAAW,EAAE,MAAM;EACnB,KAAK,EAAE,IAAI,GAcZ;EA7CH,AAiCI,4CAjCwC,AA4BzC,0DAA0D,CAKzD,KAAK,CAAC;IACJ,KAAK,EAAE,IAAI;IACX,SAAS,EAAE,CAAC;IACZ,WAAW,EAAE,CAAC;IACd,UAAU,EAAE,GAAG;IACf,YAAY,EAAE,EAAE,GACjB;EAvCL,AAyCI,4CAzCwC,AA4BzC,0DAA0D,CAazD,kBAAkB,CAAC;IACjB,SAAS,EAAE,cAAc;IACzB,KAAK,EAAE,cAAc,GACtB;;AAIL,AACE,yBADuB,AACtB,2BAA2B,CAAC;EAC3B,KAAK,EAAE,OAAO,GACf;;AAHH,AAIE,yBAJuB,AAItB,2BAA2B,CAAC;EAC3B,KAAK,EAAE,OAAO,GACf;;AANH,AAOE,yBAPuB,AAOtB,2BAA2B,CAAC;EAC3B,KAAK,EAAE,OAAO,GACf;;AATH,AAUE,yBAVuB,AAUtB,2BAA2B,CAAC;EAC3B,KAAK,EAAE,OAAO,GACf;;AAZH,AAaE,yBAbuB,AAatB,2BAA2B,CAAC;EAC3B,KAAK,EAAE,OAAO,GACf;;AAGH,AAEI,KAFC,CACH,QAAQ,CACN,CAAC,AAAA,sBAAsB,CAAC;EACtB,UAAU,EAAE,iBAAiB;EAC7B,WAAW,EAAE,KAAK;EAClB,YAAY,EAAE,KAAK;EACnB,OAAO,EAAE,MAAM;EACf,UAAU,EAAE,CAAC;EACb,aAAa,EAAE,CAAC;EAChB,YAAY,EAAE,IAAI,GAKnB;EAdL,AAWM,KAXD,CACH,QAAQ,CACN,CAAC,AAAA,sBAAsB,AASpB,QAAQ,CAAC;IACR,OAAO,EAAE,IAAI,GACd" +} \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index be73728ad..6a8eb5abf 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "woocommerce-germanized", - "version": "3.9.2", + "version": "3.11.4", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/packages/one-stop-shop-woocommerce/assets/css/admin.css b/packages/one-stop-shop-woocommerce/assets/css/admin.css new file mode 100644 index 000000000..e347a856e --- /dev/null +++ b/packages/one-stop-shop-woocommerce/assets/css/admin.css @@ -0,0 +1,282 @@ +p.oss-woocommerce-additional-desc { + margin-top: 1em !important; + line-height: 1.5em; + background: #fff; + padding: .5em; + font-style: normal; + font-size: 14px; + box-shadow: 0 1px 1px 0 rgba(0, 0, 0, 0.1); } + +h2.oss-woocommerce-settings-title { + margin-top: 1.5em; } + h2.oss-woocommerce-settings-title .page-title-action { + top: 0; } + +.oss-add-tax-class-by-country-template { + display: none; } + +a.oss-remove-tax-class-by-country { + text-indent: 0; + overflow: hidden; + color: #b32d2e; + text-decoration: none; + vertical-align: middle; } + +p.oss-tax-class-by-country-field { + display: flex; + flex-wrap: wrap; + align-items: center; } + p.oss-tax-class-by-country-field label { + width: 100%; } + p.oss-tax-class-by-country-field select { + width: auto; + flex-grow: 1; + flex-shrink: 0; + flex-basis: 50%; } + p.oss-tax-class-by-country-field a.oss-remove-tax-class-by-country { + margin-left: .5em; } + +p.oss-add-tax-class-by-country-field { + display: flex; + align-items: center; + width: 100%; } + p.oss-add-tax-class-by-country-field a.oss-remove-tax-class-by-country { + margin-left: .5em; } + p.oss-add-tax-class-by-country-field label { + width: auto; + flex-grow: 0; + flex-shrink: 0; + flex-basis: 30%; + margin-right: 1em; } + p.oss-add-tax-class-by-country-field select.oss-tax-class-new-class { + width: auto; + flex-grow: 1; + flex-shrink: 0; + flex-basis: 50%; } + +.oss-observer-total { + font-size: 1.5em; + background: #b1dabc; + color: #1d4026; + padding: 3px; + border-radius: 2px; } + .oss-observer-total.observer-total-red { + background: #dab1b4; + color: #401d1d; } + +.oss-settings-learn-more, .oss-settings-refresh-tax-rates { + margin-left: .5em; } + +.oss-observer-date-end { + color: #646970; + font-size: 12px; + margin-left: .5em; } + +.oss-woo-status { + background: #eee; + padding: .2em .5em; + font-size: .9em; + border-radius: 3px; + display: inline-flex; + white-space: nowrap; } + .oss-woo-status.report-status-pending { + background: #f8dda7; + color: #94660c; } + .oss-woo-status.report-status-failed { + background: #eba3a3; + color: #761919; } + .oss-woo-status.report-status-completed { + background: #c6e1c6; + color: #5b841b; } + +.create-oss-reports { + text-align: center; + max-width: 700px; + margin: 40px auto; } + .create-oss-reports .create-oss-report { + background: #fff; + overflow: hidden; + padding: 0; + margin: 0 0 16px; + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.13); + color: #555; + text-align: left; } + .create-oss-reports header { + border-bottom: 1px solid #eee; + margin: 0; + padding: 24px 24px 0; } + .create-oss-reports header h2 { + margin: 0 0 24px; + color: #555; + font-size: 24px; + font-weight: 400; + line-height: 1em; } + .create-oss-reports section { + padding: 24px 24px 0; } + .create-oss-reports section .oss-report-options .select2-container { + min-width: 400px; } + .create-oss-reports section .oss-report-options td, .create-oss-reports section .oss-report-options th { + vertical-align: middle; + line-height: 1.75em; + padding: 0 0 24px; } + .create-oss-reports section .oss-report-options th { + width: 25%; + padding-right: 20px; } + .create-oss-reports section .oss-report-options th label { + color: #555; + font-weight: 400; + position: relative; + display: block; } + .create-oss-reports section .oss-report-hidden { + display: none; } + .create-oss-reports .oss-actions { + overflow: hidden; + border-top: 1px solid #eee; + margin: 0; + padding: 23px 24px 24px; + line-height: 3em; + display: flex; + flex-wrap: wrap; + justify-content: flex-end; } + .create-oss-reports .oss-actions .button { + font-size: 1.25em; + padding: 0.5em 1em !important; + line-height: 1.5em !important; + margin-right: .5em; + margin-bottom: 2px; + height: auto !important; + border-radius: 4px; + opacity: 1; } + +.woocommerce_page_oss-reports .summary { + font-family: HelveticaNeue-Light,"Helvetica Neue Light","Helvetica Neue",sans-serif; + font-weight: 400; + line-height: 1.6em; + font-size: 16px; } + +.woocommerce_page_oss-reports .tablenav .actions { + overflow: visible; } + +.woocommerce_page_oss-reports .tablenav .select2-container { + float: left; + width: 240px !important; + font-size: 14px; + vertical-align: middle; + margin: 1px 6px 4px 1px; } + .woocommerce_page_oss-reports .tablenav .select2-container .select2-selection--single { + height: 32px; } + .woocommerce_page_oss-reports .tablenav .select2-container .select2-selection--single .select2-selection__rendered { + line-height: 29px; } + .woocommerce_page_oss-reports .tablenav .select2-container .select2-selection--single .select2-selection__arrow { + height: 30px; } + +.woocommerce_page_oss-reports .tablenav select, .woocommerce_page_oss-reports .tablenav input { + line-height: 1; + height: 32px; } + +.woocommerce_page_oss-reports .tablenav input { + height: 31px; } + +.woocommerce_page_oss-reports .wp-list-table { + margin-top: 1em; } + .woocommerce_page_oss-reports .wp-list-table td, .woocommerce_page_oss-reports .wp-list-table th { + padding: .5em 1em; + width: 10ch; + vertical-align: middle; } + .woocommerce_page_oss-reports .wp-list-table td, .woocommerce_page_oss-reports .wp-list-table tbody th { + line-height: 26px; } + .woocommerce_page_oss-reports .wp-list-table thead th { + padding: .5em 1em; } + .woocommerce_page_oss-reports .wp-list-table thead th.sortable a, .woocommerce_page_oss-reports .wp-list-table thead th.sorted a { + padding: 0; } + .woocommerce_page_oss-reports .wp-list-table thead th:last-child { + padding-right: 2em; } + .woocommerce_page_oss-reports .wp-list-table .check-column { + width: 16px; + white-space: nowrap; + padding: 1em 1em 1em 1em !important; + vertical-align: middle; } + .woocommerce_page_oss-reports .wp-list-table .check-column input { + vertical-align: text-top; + margin: 1px 0; } + .woocommerce_page_oss-reports .wp-list-table td.column-title { + font-weight: bold; } + .woocommerce_page_oss-reports .wp-list-table .column-title { + width: 20ch; } + .woocommerce_page_oss-reports .wp-list-table .column-actions { + width: 10ch; } + .woocommerce_page_oss-reports .wp-list-table .column-actions { + text-align: right; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button { + text-indent: 9999px; + margin: 2px 0 2px 4px; + position: relative; + display: inline-block; + padding: 0; + height: 2em; + width: 2em; + overflow: hidden; + vertical-align: middle; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button::after { + font-family: Dashicons; + margin: 0; + margin-top: 2px; + speak: none; + font-weight: 400; + font-variant: normal; + text-transform: none; + text-indent: 0; + position: absolute; + top: 0; + left: 0; + width: 100%; + height: 100%; + text-align: center; + line-height: 1.85; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.view::after { + font-family: WooCommerce; + content: "\e010"; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.refresh::after { + content: "\f515"; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete, .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel { + border-color: #a00; + color: #a00; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete:focus, .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel:focus { + box-shadow: 0 0 0 1px #a00; + border-color: #a00; + color: #a00; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete:hover, .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel:hover { + border-color: #910000; + color: #910000; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete::after, .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel::after { + font-family: Dashicons; + content: "\f182"; } + .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.export::after, .woocommerce_page_oss-reports .wp-list-table .column-actions a.button.export_bop::after { + content: "\f103"; } + .woocommerce_page_oss-reports .wp-list-table .column-address, .woocommerce_page_oss-reports .wp-list-table .column-sender { + width: 20ch; } + .woocommerce_page_oss-reports .wp-list-table .column-items { + width: 20ch; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview { + font-size: .9em; + border-spacing: 0; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview thead th { + color: #adadad; + padding-top: 0; + font-size: 1.1em; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview tr td { + border-bottom: 1px solid #ccc !important; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview tr:last-child td { + border-bottom: none !important; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th, .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td { + padding: .3em 0; + vertical-align: top; + line-height: 20px; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th.wc-gzd-shipment-item-column-name, .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td.wc-gzd-shipment-item-column-name { + width: 70%; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th.wc-gzd-shipment-item-column-name small, .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td.wc-gzd-shipment-item-column-name small { + color: #999; + font-size: 12px; } + .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th.wc-gzd-shipment-item-column-quantity, .woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td.wc-gzd-shipment-item-column-quantity { + text-align: right; + padding-right: .5em; } \ No newline at end of file diff --git a/packages/one-stop-shop-woocommerce/assets/css/admin.min.css b/packages/one-stop-shop-woocommerce/assets/css/admin.min.css new file mode 100644 index 000000000..bb53cb729 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/assets/css/admin.min.css @@ -0,0 +1 @@ +p.oss-woocommerce-additional-desc{margin-top:1em!important;line-height:1.5em;background:#fff;padding:.5em;font-style:normal;font-size:14px;box-shadow:0 1px 1px 0 rgba(0,0,0,.1)}h2.oss-woocommerce-settings-title{margin-top:1.5em}h2.oss-woocommerce-settings-title .page-title-action{top:0}.oss-add-tax-class-by-country-template{display:none}a.oss-remove-tax-class-by-country{text-indent:0;overflow:hidden;color:#b32d2e;text-decoration:none;vertical-align:middle}p.oss-tax-class-by-country-field{display:flex;flex-wrap:wrap;align-items:center}p.oss-tax-class-by-country-field label{width:100%}p.oss-tax-class-by-country-field select{width:auto;flex-grow:1;flex-shrink:0;flex-basis:50%}p.oss-tax-class-by-country-field a.oss-remove-tax-class-by-country{margin-left:.5em}p.oss-add-tax-class-by-country-field{display:flex;align-items:center;width:100%}p.oss-add-tax-class-by-country-field a.oss-remove-tax-class-by-country{margin-left:.5em}p.oss-add-tax-class-by-country-field label{width:auto;flex-grow:0;flex-shrink:0;flex-basis:30%;margin-right:1em}p.oss-add-tax-class-by-country-field select.oss-tax-class-new-class{width:auto;flex-grow:1;flex-shrink:0;flex-basis:50%}.oss-observer-total{font-size:1.5em;background:#b1dabc;color:#1d4026;padding:3px;border-radius:2px}.oss-observer-total.observer-total-red{background:#dab1b4;color:#401d1d}.oss-settings-learn-more,.oss-settings-refresh-tax-rates{margin-left:.5em}.oss-observer-date-end{color:#646970;font-size:12px;margin-left:.5em}.oss-woo-status{background:#eee;padding:.2em .5em;font-size:.9em;border-radius:3px;display:inline-flex;white-space:nowrap}.oss-woo-status.report-status-pending{background:#f8dda7;color:#94660c}.oss-woo-status.report-status-failed{background:#eba3a3;color:#761919}.oss-woo-status.report-status-completed{background:#c6e1c6;color:#5b841b}.create-oss-reports{text-align:center;max-width:700px;margin:40px auto}.create-oss-reports .create-oss-report{background:#fff;overflow:hidden;padding:0;margin:0 0 16px;box-shadow:0 1px 3px rgba(0,0,0,.13);color:#555;text-align:left}.create-oss-reports header{border-bottom:1px solid #eee;margin:0;padding:24px 24px 0}.create-oss-reports header h2{margin:0 0 24px;color:#555;font-size:24px;font-weight:400;line-height:1em}.create-oss-reports section{padding:24px 24px 0}.create-oss-reports section .oss-report-options .select2-container{min-width:400px}.create-oss-reports section .oss-report-options td,.create-oss-reports section .oss-report-options th{vertical-align:middle;line-height:1.75em;padding:0 0 24px}.create-oss-reports section .oss-report-options th{width:25%;padding-right:20px}.create-oss-reports section .oss-report-options th label{color:#555;font-weight:400;position:relative;display:block}.create-oss-reports section .oss-report-hidden{display:none}.create-oss-reports .oss-actions{overflow:hidden;border-top:1px solid #eee;margin:0;padding:23px 24px 24px;line-height:3em;display:flex;flex-wrap:wrap;justify-content:flex-end}.create-oss-reports .oss-actions .button{font-size:1.25em;padding:.5em 1em!important;line-height:1.5em!important;margin-right:.5em;margin-bottom:2px;height:auto!important;border-radius:4px;opacity:1}.woocommerce_page_oss-reports .summary{font-family:HelveticaNeue-Light,"Helvetica Neue Light","Helvetica Neue",sans-serif;font-weight:400;line-height:1.6em;font-size:16px}.woocommerce_page_oss-reports .tablenav .actions{overflow:visible}.woocommerce_page_oss-reports .tablenav .select2-container{float:left;width:240px!important;font-size:14px;vertical-align:middle;margin:1px 6px 4px 1px}.woocommerce_page_oss-reports .tablenav .select2-container .select2-selection--single{height:32px}.woocommerce_page_oss-reports .tablenav .select2-container .select2-selection--single .select2-selection__rendered{line-height:29px}.woocommerce_page_oss-reports .tablenav .select2-container .select2-selection--single .select2-selection__arrow{height:30px}.woocommerce_page_oss-reports .tablenav input,.woocommerce_page_oss-reports .tablenav select{line-height:1;height:32px}.woocommerce_page_oss-reports .tablenav input{height:31px}.woocommerce_page_oss-reports .wp-list-table{margin-top:1em}.woocommerce_page_oss-reports .wp-list-table td,.woocommerce_page_oss-reports .wp-list-table th{padding:.5em 1em;width:10ch;vertical-align:middle}.woocommerce_page_oss-reports .wp-list-table tbody th,.woocommerce_page_oss-reports .wp-list-table td{line-height:26px}.woocommerce_page_oss-reports .wp-list-table thead th{padding:.5em 1em}.woocommerce_page_oss-reports .wp-list-table thead th.sortable a,.woocommerce_page_oss-reports .wp-list-table thead th.sorted a{padding:0}.woocommerce_page_oss-reports .wp-list-table thead th:last-child{padding-right:2em}.woocommerce_page_oss-reports .wp-list-table .check-column{width:16px;white-space:nowrap;padding:1em 1em 1em 1em!important;vertical-align:middle}.woocommerce_page_oss-reports .wp-list-table .check-column input{vertical-align:text-top;margin:1px 0}.woocommerce_page_oss-reports .wp-list-table td.column-title{font-weight:700}.woocommerce_page_oss-reports .wp-list-table .column-title{width:20ch}.woocommerce_page_oss-reports .wp-list-table .column-actions{width:10ch}.woocommerce_page_oss-reports .wp-list-table .column-actions{text-align:right}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button{text-indent:9999px;margin:2px 0 2px 4px;position:relative;display:inline-block;padding:0;height:2em;width:2em;overflow:hidden;vertical-align:middle}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button::after{font-family:Dashicons;margin:0;margin-top:2px;speak:none;font-weight:400;font-variant:normal;text-transform:none;text-indent:0;position:absolute;top:0;left:0;width:100%;height:100%;text-align:center;line-height:1.85}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.view::after{font-family:WooCommerce;content:"\e010"}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.refresh::after{content:"\f515"}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel,.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete{border-color:#a00;color:#a00}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel:focus,.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete:focus{box-shadow:0 0 0 1px #a00;border-color:#a00;color:#a00}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel:hover,.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete:hover{border-color:#910000;color:#910000}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.cancel::after,.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.delete::after{font-family:Dashicons;content:"\f182"}.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.export::after,.woocommerce_page_oss-reports .wp-list-table .column-actions a.button.export_bop::after{content:"\f103"}.woocommerce_page_oss-reports .wp-list-table .column-address,.woocommerce_page_oss-reports .wp-list-table .column-sender{width:20ch}.woocommerce_page_oss-reports .wp-list-table .column-items{width:20ch}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview{font-size:.9em;border-spacing:0}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview thead th{color:#adadad;padding-top:0;font-size:1.1em}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview tr td{border-bottom:1px solid #ccc!important}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview tr:last-child td{border-bottom:none!important}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td,.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th{padding:.3em 0;vertical-align:top;line-height:20px}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td.wc-gzd-shipment-item-column-name,.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th.wc-gzd-shipment-item-column-name{width:70%}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td.wc-gzd-shipment-item-column-name small,.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th.wc-gzd-shipment-item-column-name small{color:#999;font-size:12px}.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview td.wc-gzd-shipment-item-column-quantity,.woocommerce_page_oss-reports .wp-list-table #the-list .column-items table.wc-gzd-shipments-preview th.wc-gzd-shipment-item-column-quantity{text-align:right;padding-right:.5em} \ No newline at end of file diff --git a/packages/one-stop-shop-woocommerce/assets/css/admin.scss b/packages/one-stop-shop-woocommerce/assets/css/admin.scss new file mode 100644 index 000000000..ff78032ad --- /dev/null +++ b/packages/one-stop-shop-woocommerce/assets/css/admin.scss @@ -0,0 +1,427 @@ +p.oss-woocommerce-additional-desc { + margin-top: 1em !important; + line-height: 1.5em; + background: #fff; + padding: .5em; + font-style: normal; + font-size: 14px; + box-shadow: 0 1px 1px 0 rgba(0, 0, 0, 0.1); +} + +h2.oss-woocommerce-settings-title { + margin-top: 1.5em; + + .page-title-action { + top: 0; + } +} + +.oss-add-tax-class-by-country-template { + display: none; +} + +a.oss-remove-tax-class-by-country { + text-indent: 0; + overflow: hidden; + color: #b32d2e; + text-decoration: none; + vertical-align: middle; +} + +p.oss-tax-class-by-country-field { + display: flex; + flex-wrap: wrap; + align-items: center; + + label { + width: 100%; + } + + select { + width: auto; + flex-grow: 1; + flex-shrink: 0; + flex-basis: 50%; + } + + a.oss-remove-tax-class-by-country { + margin-left: .5em; + } +} + +p.oss-add-tax-class-by-country-field { + display: flex; + align-items: center; + width: 100%; + + a.oss-remove-tax-class-by-country { + margin-left: .5em; + } + + label { + width: auto; + flex-grow: 0; + flex-shrink: 0; + flex-basis: 30%; + margin-right: 1em; + } + + select.oss-tax-class-new-class { + width: auto; + flex-grow: 1; + flex-shrink: 0; + flex-basis: 50%; + } +} + +.oss-observer-total { + font-size: 1.5em; + background: #b1dabc; + color: #1d4026; + padding: 3px; + border-radius: 2px; + + &.observer-total-red { + background: #dab1b4; + color: #401d1d; + } +} + +.oss-settings-learn-more, .oss-settings-refresh-tax-rates { + margin-left: .5em; +} + +.oss-observer-date-end { + color: #646970; + font-size: 12px; + margin-left: .5em; +} + +.oss-woo-status { + background: #eee; + padding: .2em .5em; + font-size: .9em; + border-radius: 3px; + display: inline-flex; + white-space: nowrap; + + &.report-status-pending { + background: #f8dda7; + color: #94660c; + } + + &.report-status-failed { + background: #eba3a3; + color: #761919; + } + + &.report-status-completed { + background: #c6e1c6; + color: #5b841b; + } +} + +.create-oss-reports { + text-align: center; + max-width: 700px; + margin: 40px auto; + + .create-oss-report { + background: #fff; + overflow: hidden; + padding: 0; + margin: 0 0 16px; + box-shadow: 0 1px 3px rgba(0,0,0,.13); + color: #555; + text-align: left; + } + + header { + border-bottom: 1px solid #eee; + margin: 0; + padding: 24px 24px 0; + + h2 { + margin: 0 0 24px; + color: #555; + font-size: 24px; + font-weight: 400; + line-height: 1em; + } + } + + section { + padding: 24px 24px 0; + + .oss-report-options { + .select2-container { + min-width: 400px; + } + + td, th { + vertical-align: middle; + line-height: 1.75em; + padding: 0 0 24px; + } + + th { + width: 25%; + padding-right: 20px; + + label { + color: #555; + font-weight: 400; + position: relative; + display: block; + } + } + } + + .oss-report-hidden { + display: none; + } + } + + .oss-actions { + overflow: hidden; + border-top: 1px solid #eee; + margin: 0; + padding: 23px 24px 24px; + line-height: 3em; + display: flex; + flex-wrap: wrap; + justify-content: flex-end; + + .button { + font-size: 1.25em; + padding: .5em 1em!important; + line-height: 1.5em!important; + margin-right: .5em; + margin-bottom: 2px; + height: auto!important; + border-radius: 4px; + opacity: 1; + } + } +} + +.woocommerce_page_oss-reports { + .summary { + font-family: HelveticaNeue-Light,"Helvetica Neue Light","Helvetica Neue",sans-serif; + font-weight: 400; + line-height: 1.6em; + font-size: 16px; + } + + .tablenav { + .actions { + overflow: visible; + } + + .select2-container { + float: left; + width: 240px!important; + font-size: 14px; + vertical-align: middle; + margin: 1px 6px 4px 1px; + + .select2-selection--single { + height: 32px; + + .select2-selection__rendered { + line-height: 29px; + } + + .select2-selection__arrow { + height: 30px; + } + } + } + + select, input { + line-height: 1; + height: 32px; + } + + input { + height: 31px; + } + } + + .wp-list-table { + margin-top: 1em; + + td, th { + padding: .5em 1em; + width: 10ch; + vertical-align: middle; + } + + td, tbody th { + line-height: 26px; + } + + thead { + th { + padding: .5em 1em; + + &.sortable a, &.sorted a { + padding: 0; + } + + &:last-child { + padding-right: 2em; + } + } + } + + .check-column { + width: 16px; + white-space: nowrap; + padding: 1em 1em 1em 1em !important; + vertical-align: middle; + + input { + vertical-align: text-top; + margin: 1px 0; + } + } + + td.column-title { + font-weight: bold; + } + + .column-title { + width: 20ch; + } + + .column-actions { + width: 10ch; + } + + .column-actions { + text-align: right; + + a.button { + text-indent: 9999px; + margin: 2px 0 2px 4px; + position: relative; + display: inline-block; + padding: 0; + height: 2em; + width: 2em; + overflow: hidden; + vertical-align: middle; + + &::after { + font-family: Dashicons; + margin: 0; + margin-top: 2px; + speak: none; + font-weight: 400; + font-variant: normal; + text-transform: none; + text-indent: 0; + position: absolute; + top: 0; + left: 0; + width: 100%; + height: 100%; + text-align: center; + line-height: 1.85; + } + + &.view::after { + font-family: WooCommerce; + content: "\e010"; + } + + &.refresh::after { + content: "\f515"; + } + + &.delete, &.cancel { + border-color: #a00; + color: #a00; + + &:focus { + box-shadow: 0 0 0 1px #a00; + border-color: #a00; + color: #a00; + } + + &:hover { + border-color: darken( #a00, 5% ); + color: darken( #a00, 5% ); + } + + &::after { + font-family: Dashicons; + content: "\f182"; + } + } + + &.export::after, &.export_bop::after { + content: "\f103"; + } + } + } + + .column-address, .column-sender { + width: 20ch; + } + + .column-items { + width: 20ch; + } + + #the-list { + .column-items { + table.wc-gzd-shipments-preview { + font-size: .9em; + border-spacing: 0; + + thead { + th { + color: #adadad; + padding-top: 0; + font-size: 1.1em; + } + } + + tr { + td { + border-bottom: 1px solid #ccc !important; + } + + &:last-child { + td { + border-bottom: none !important; + } + } + } + + th, td { + padding: .3em 0; + vertical-align: top; + line-height: 20px; + + &.wc-gzd-shipment-item-column-name { + width: 70%; + + small { + color: #999; + font-size: 12px; + } + } + + &.wc-gzd-shipment-item-column-quantity { + text-align: right; + padding-right: .5em; + } + } + } + } + } + } +} \ No newline at end of file diff --git a/packages/one-stop-shop-woocommerce/assets/js/admin.js b/packages/one-stop-shop-woocommerce/assets/js/admin.js new file mode 100644 index 000000000..2ede63f8f --- /dev/null +++ b/packages/one-stop-shop-woocommerce/assets/js/admin.js @@ -0,0 +1,104 @@ +window.oss = window.oss || {}; + +( function( $, oss ) { + oss.admin = { + + params: {}, + dates: false, + + init: function() { + var self = oss.admin; + self.params = oss_admin_params; + + $( document ) + .on( 'change', 'select#oss-report-type', self.onChangeReportType ) + .on( 'click', 'a.oss-add-new-tax-class-by-country', self.onAddNewTaxClassCountry ) + .on( 'click', 'a.oss-remove-tax-class-by-country', self.onRemoveTaxClassCountry ); + + if ( $( 'select#oss-report-type' ).length > 0 ) { + $( 'select#oss-report-type' ).trigger( 'change' ); + } + + if ( $( '.oss_range_datepicker' ).length > 0 ) { + self.initDatePicker(); + } + + $( document.body ).on( 'init_tooltips', function() { + self.initTipTip(); + }); + + self.initTipTip(); + }, + + onAddNewTaxClassCountry: function() { + var $parent = $( this ).parents( '#general_product_data' ); + + if ( $parent.length === 0 ) { + $parent = $( this ).parents( '.woocommerce_variable_attributes' ); + } + + var $template = $parent.find( '.oss-add-tax-class-by-country-template:first' ).clone(); + + $template.removeClass( 'oss-add-tax-class-by-country-template' ).addClass( 'oss-add-tax-class-by-country-new' ); + $parent.find( '.oss-new-tax-class-by-country-placeholder' ).append( $template ).show(); + + return false; + }, + + onRemoveTaxClassCountry: function() { + var $parent = $( this ).parents( '.form-field' ); + + // Trigger change to notify Woo about an update (variations). + $parent.find( 'select' ).trigger( 'change' ); + $parent.remove(); + + return false; + }, + + initDatePicker: function() { + var self = oss.admin; + + self.dates = $( '.oss_range_datepicker' ).datepicker({ + changeMonth: true, + changeYear: true, + defaultDate: '', + dateFormat: 'yy-mm-dd', + numberOfMonths: 1, + minDate: '-20Y', + maxDate: '+0D', + showButtonPanel: true, + showOn: 'focus', + buttonImageOnly: true, + onSelect: function() { + var option = $( this ).is( '.from' ) ? 'minDate' : 'maxDate', + date = $( this ).datepicker( 'getDate' ); + + self.dates.not( this ).datepicker( 'option', option, date ); + } + }); + }, + + onChangeReportType: function() { + var type = $( this ).val(); + + $( '.oss-report-hidden' ).hide(); + + if ( $( '.oss-report-' + type ).length > 0 ) { + $( '.oss-report-' + type ).show(); + } + }, + + initTipTip: function() { + $( '.column-actions .oss-woo-action-button' ).tipTip( { + 'fadeIn': 50, + 'fadeOut': 50, + 'delay': 200 + }); + } + }; + + $( document ).ready( function() { + oss.admin.init(); + }); + +})( jQuery, window.oss ); diff --git a/packages/one-stop-shop-woocommerce/assets/js/admin.min.js b/packages/one-stop-shop-woocommerce/assets/js/admin.min.js new file mode 100644 index 000000000..cadebe2f0 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/assets/js/admin.min.js @@ -0,0 +1 @@ +window.oss=window.oss||{},function(a,e){e.admin={params:{},dates:!1,init:function(){var t=e.admin;t.params=oss_admin_params,a(document).on("change","select#oss-report-type",t.onChangeReportType).on("click","a.oss-add-new-tax-class-by-country",t.onAddNewTaxClassCountry).on("click","a.oss-remove-tax-class-by-country",t.onRemoveTaxClassCountry),0\n" +"Language-Team: \n" +"Language: de_DE\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Generator: Poedit 3.1.1\n" +"X-Poedit-Basepath: ../..\n" +"X-Poedit-Flags-xgettext: --add-comments=translators:\n" +"X-Poedit-WPHeader: one-stop-shop-woocommerce.php\n" +"X-Poedit-SourceCharset: UTF-8\n" +"X-Poedit-KeywordsList: __;_e;_n:1,2;_x:1,2c;_ex:1,2c;_nx:4c,1,2;esc_attr__;" +"esc_attr_e;esc_attr_x:1,2c;esc_html__;esc_html_e;esc_html_x:1,2c;_n_noop:1,2;" +"_nx_noop:3c,1,2;__ngettext_noop:1,2\n" +"X-Poedit-SearchPath-0: .\n" +"X-Poedit-SearchPathExcluded-0: *.min.js\n" +"X-Poedit-SearchPathExcluded-1: vendor\n" +"X-Poedit-SearchPathExcluded-2: node_modules\n" +"X-Poedit-SearchPathExcluded-3: build\n" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:312 +msgid "Reduced rate" +msgstr "" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:313 +msgctxt "tax-helper-tax-class-name" +msgid "Greater reduced rate" +msgstr "Zusätzlicher reduzierter Preis" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:314 +msgctxt "tax-helper-tax-class-name" +msgid "Super reduced rate" +msgstr "Stark reduzierter Preis" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:718 +msgctxt "tax-helper" +msgid "Madeira" +msgstr "Madeira" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:725 +msgctxt "tax-helper" +msgid "Acores" +msgstr "Azoren" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:761 +msgctxt "tax-helper" +msgid "Northern Ireland" +msgstr "Nordirland" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:774 +msgctxt "tax-helper-rate-import" +msgid "Exempt" +msgstr "Ausnahme" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:843 +#, php-format +msgctxt "tax-helper-rate-import" +msgid "VAT %1$s %% %2$s" +msgstr "MwSt. %1$s %% %2$s" + +#. translators: 1: composer command. 2: plugin directory +#: one-stop-shop-woocommerce.php:44 one-stop-shop-woocommerce.php:62 +#, php-format +msgctxt "oss" +msgid "" +"Your installation of the One Stop Shop feature plugin is incomplete. Please " +"run %1$s within the %2$s directory." +msgstr "" +"Deine Installation des One Stop Shop Feature Plugins ist nicht komplett. " +"Bitte führe %1$s innerhalb des %2$s Verzeichnisses aus." + +#: src/Admin.php:78 src/Admin.php:79 +msgctxt "oss" +msgid "Refresh VAT rates (OSS)" +msgstr "EU Steuersätze erneuern (OSS)" + +#: src/Admin.php:83 +msgctxt "oss" +msgid "Note:" +msgstr "Hinweis:" + +#: src/Admin.php:84 +#, php-format +msgctxt "oss" +msgid "" +"This option will delete all of your current EU VAT rates and re-import them " +"based on your current OSS status." +msgstr "" +"Diese Option löscht alle deine aktuellen EU Steuersätze und importiert sie " +"anschließend auf Basis deines OSS Status neu." + +#: src/Admin.php:214 +#, php-format +msgctxt "oss" +msgid "" +"Seems like you have reached (or are close to reaching) the delivery " +"threshold for the current year. Please make sure to check the report details and take action in case necessary." +msgstr "" +"Du bist kurz davor oder hast bereits die Lieferschwelle für das aktuelle " +"Jahr überschritten. Bitte prüfe die Details " +"des Berichts und unternimm weitere Schritte." + +#: src/Admin.php:218 +msgctxt "oss" +msgid "Delivery threshold reached (OSS)" +msgstr "Lieferschwelle erreicht (OSS)" + +#: src/Admin.php:448 src/SettingsPage.php:17 +msgctxt "oss" +msgid "OSS" +msgstr "OSS" + +#: src/Admin.php:448 src/Admin.php:566 src/SettingsPage.php:23 +msgctxt "oss" +msgid "One Stop Shop" +msgstr "One Stop Shop" + +#: src/Admin.php:469 src/Package.php:365 +#, php-format +msgctxt "oss" +msgid "Q%1$s/%2$s" +msgstr "Q%1$s/%2$s" + +#: src/Admin.php:478 src/Package.php:370 +#, php-format +msgctxt "oss" +msgid "%1$s/%2$s" +msgstr "%1$s/%2$s" + +#: src/Admin.php:486 +msgctxt "oss" +msgid "New Report" +msgstr "Neuer Bericht" + +#: src/Admin.php:493 +msgctxt "oss" +msgid "Type" +msgstr "Typ" + +#: src/Admin.php:505 +msgctxt "oss" +msgid "Year" +msgstr "Jahr" + +#: src/Admin.php:517 +msgctxt "oss" +msgid "Quarter" +msgstr "Quartal" + +#: src/Admin.php:529 +msgctxt "oss" +msgid "Month" +msgstr "Monat" + +#: src/Admin.php:541 +msgctxt "oss" +msgid "Date range" +msgstr "Zeitraum" + +#: src/Admin.php:553 +msgctxt "oss" +msgid "Start report" +msgstr "Bericht starten" + +#: src/Admin.php:567 +msgctxt "oss" +msgid "New report" +msgstr "Neuer Bericht" + +#: src/Admin.php:609 +msgctxt "oss" +msgid "View" +msgstr "Ansehen" + +#: src/Admin.php:613 +msgctxt "oss" +msgid "Export" +msgstr "Exportieren" + +#: src/Admin.php:617 +msgctxt "oss" +msgid "Export BOP" +msgstr "BOP-CSV exportieren" + +#: src/Admin.php:621 +msgctxt "oss" +msgid "Refresh" +msgstr "Aktualisieren" + +#: src/Admin.php:625 +msgctxt "oss" +msgid "Delete" +msgstr "Löschen" + +#: src/Admin.php:631 +msgctxt "oss" +msgid "Cancel" +msgstr "Abbrechen" + +#: src/Admin.php:669 +msgctxt "oss" +msgid "Country" +msgstr "Land" + +#: src/Admin.php:670 +msgctxt "oss" +msgid "Tax Rate" +msgstr "Steuersatz" + +#: src/Admin.php:671 +msgctxt "oss" +msgid "Net Total" +msgstr "Nettobetrag" + +#: src/Admin.php:672 +msgctxt "oss" +msgid "Tax Total" +msgstr "Steuerbertrag" + +#: src/Admin.php:703 +#, php-format +msgctxt "oss" +msgid "%1$s %%" +msgstr "%1$s %%" + +#: src/Admin.php:716 +#, php-format +msgctxt "oss" +msgid "" +"Currently processed %1$s orders. Next iteration is scheduled for %2$s. Find pending actions" +msgstr "" +"Aktuell %1$s Bestellung verarbeitet. Nächster Durchlauf planmäßig am %2$s. " +"Offene Aktionen finden" + +#: src/Admin.php:716 +msgctxt "oss" +msgid "Not yet known" +msgstr "Noch nicht bekannt" + +#: src/Admin.php:747 src/ReportTable.php:39 src/SettingsPage.php:23 +msgctxt "oss" +msgid "Reports" +msgstr "Berichte" + +#: src/AdminNote.php:46 +msgctxt "oss" +msgid "Dismiss" +msgstr "Ausblenden" + +#: src/AsyncReportGenerator.php:258 +msgctxt "oss" +msgid "No orders found." +msgstr "Keine Bestellungen gefunden." + +#: src/CSVExporter.php:54 +msgctxt "oss" +msgid "Country code" +msgstr "Land des Verbrauchs" + +#: src/CSVExporter.php:55 +msgctxt "oss" +msgid "Tax rate" +msgstr "Umsatzsteuersatz" + +#: src/CSVExporter.php:56 +msgctxt "oss" +msgid "Taxable base" +msgstr "Nettobetrag" + +#: src/CSVExporter.php:57 +msgctxt "oss" +msgid "Amount" +msgstr "Umsatzsteuerbetrag" + +#: src/DeliveryThresholdEmailNotification.php:19 +msgctxt "oss" +msgid "OSS Delivery Threshold Notification" +msgstr "OSS Lieferschwelle Benachrichtigung" + +#: src/DeliveryThresholdEmailNotification.php:20 +msgctxt "oss" +msgid "" +"This email notifies shop owners in case the delivery threshold (OSS) is " +"close to being reached." +msgstr "" +"Diese E-Mail benachrichtigt den Shopbetreiber über eine in Kürze anstehende " +"Überschreitung der Lieferschwelle (OSS)." + +#: src/DeliveryThresholdEmailNotification.php:38 +msgctxt "oss" +msgid "[{site_title}]: OSS delivery threshold reached" +msgstr "[{site_title}]: OSS Lieferschwelle erreicht" + +#: src/DeliveryThresholdEmailNotification.php:48 +msgctxt "oss" +msgid "OSS delivery threshold reached" +msgstr "OSS Lieferschwelle erreicht" + +#: src/DeliveryThresholdWarning.php:14 +msgctxt "oss" +msgid "See details" +msgstr "Details ansehen" + +#: src/Package.php:178 +msgctxt "oss" +msgid "" +"To use the OSS for WooCommerce plugin please make sure that WooCommerce is " +"installed and activated." +msgstr "" +"Um das OSS für WooCommerce Plugin nutzen zu können muss WooCommerce " +"installiert und aktiviert sein." + +#: src/Package.php:350 src/ReportTable.php:40 +msgctxt "oss" +msgid "Report" +msgstr "Bericht" + +#: src/Package.php:374 +#, php-format +msgctxt "oss" +msgid "%1$s" +msgstr "%1$s" + +#: src/Package.php:379 +#, php-format +msgctxt "oss" +msgid "%1$s - %2$s" +msgstr "%1$s - %2$s" + +#: src/Package.php:384 +#, php-format +msgctxt "oss" +msgid "Observer %1$s" +msgstr "Beobachter %1$s" + +#: src/Package.php:636 +msgctxt "oss" +msgid "Quarterly" +msgstr "Quartalsweise" + +#: src/Package.php:637 +msgctxt "oss" +msgid "Yearly" +msgstr "Jährlich" + +#: src/Package.php:638 +msgctxt "oss" +msgid "Monthly" +msgstr "Monatlich" + +#: src/Package.php:639 +msgctxt "oss" +msgid "Custom" +msgstr "Individuell" + +#: src/Package.php:643 +msgctxt "oss" +msgid "Observer" +msgstr "Beobachter" + +#: src/Package.php:657 +msgctxt "oss" +msgid "Pending" +msgstr "In Bearbeitung" + +#: src/Package.php:658 +msgctxt "oss" +msgid "Completed" +msgstr "Fertiggestellt" + +#: src/Package.php:659 +msgctxt "oss" +msgid "Failed" +msgstr "Fehlgeschlagen" + +#: src/ReportTable.php:124 +#, php-format +msgctxt "oss" +msgid "%d report deleted." +msgid_plural "%d reports deleted." +msgstr[0] "%d Bericht gelöscht." +msgstr[1] "%d Berichte gelöscht." + +#: src/ReportTable.php:187 +msgctxt "oss" +msgid "No reports found" +msgstr "Keine Berichte gefunden" + +#: src/ReportTable.php:228 +#, php-format +msgctxt "oss" +msgid "All (%s)" +msgid_plural "All (%s)" +msgstr[0] "Alle (%s)" +msgstr[1] "Alle (%s)" + +#: src/ReportTable.php:255 +#, php-format +msgctxt "oss" +msgid " (%s)" +msgid_plural " (%s)" +msgstr[0] " (%s)" +msgstr[1] " (%s)" + +#: src/ReportTable.php:326 +msgctxt "oss" +msgid "Filter" +msgstr "Filtern" + +#: src/ReportTable.php:353 +msgctxt "oss" +msgid "Title" +msgstr "Titel" + +#: src/ReportTable.php:354 +msgctxt "oss" +msgid "Start" +msgstr "Start" + +#: src/ReportTable.php:355 +msgctxt "oss" +msgid "End" +msgstr "Ende" + +#: src/ReportTable.php:356 templates/emails/admin-delivery-threshold.php:25 +msgctxt "oss" +msgid "Net total" +msgstr "Nettobetrag" + +#: src/ReportTable.php:357 templates/emails/admin-delivery-threshold.php:26 +msgctxt "oss" +msgid "Tax total" +msgstr "Steuerbertrag" + +#: src/ReportTable.php:358 +msgctxt "oss" +msgid "Status" +msgstr "Status" + +#: src/ReportTable.php:359 +msgctxt "oss" +msgid "Actions" +msgstr "Aktionen" + +#: src/ReportTable.php:423 +#, php-format +msgctxt "oss" +msgid "Select %s" +msgstr "%s auswählen" + +#: src/ReportTable.php:516 +msgctxt "oss" +msgid "Delete Permanently" +msgstr "Unwiderruflich löschen" + +#: src/Settings.php:16 +msgctxt "oss" +msgid "General" +msgstr "Allgemein" + +#: src/Settings.php:21 +msgctxt "oss" +msgid "" +"Find useful options regarding the One Stop Shop procedure here." +msgstr "" +"Finde hier nützliche Optionen zum One Stop Shop Verfahren." + +#: src/Settings.php:38 +msgctxt "oss" +msgid "OSS status" +msgstr "OSS Status" + +#: src/Settings.php:39 +msgctxt "oss" +msgid "Yes, I'm currently participating in the OSS procedure." +msgstr "Ja, ich nehme aktuell am One Stop Shop Verfahren teil." + +#: src/Settings.php:46 +msgctxt "oss" +msgid "Observation" +msgstr "Überwachung" + +#: src/Settings.php:47 +msgctxt "oss" +msgid "Automatically observe the delivery threshold of the current year." +msgstr "Überwache die Lieferschwelle des aktuellen Jahres automatisch." + +#: src/Settings.php:47 +msgctxt "oss" +msgid "" +"This option will automatically calculate the amount applicable for the OSS " +"procedure delivery threshold once per day for the current year. The report " +"will only recalculated for the days which are not yet subject to the " +"observation to save processing time." +msgstr "" +"Diese Option überwacht automatisch die Lieferschwelle des OSS Verfahrens für " +"das aktuelle Jahr, indem der dazugehörige Bericht täglich erweitert und " +"nachberechnet wird. Der Bericht wird nur für die Tage nachberechnet, für die " +"aktuell noch keine Überwachung stattfindet." + +#: src/Settings.php:59 +msgctxt "oss" +msgid "Delivery threshold" +msgstr "Lieferschwelle" + +#: src/Settings.php:72 +msgctxt "oss" +msgid "Participation" +msgstr "Teilnahme" + +#: src/Settings.php:79 +msgctxt "oss" +msgid "Report Order Date" +msgstr "Bericht Bestelldatum" + +#: src/Settings.php:80 +msgctxt "oss" +msgid "" +"Select the relevant order date to be used to determine whether to include an " +"order in a report." +msgstr "" +"Wähle das relevante Bestelldatum aus auf Basis dessen entschieden wird ob " +"eine Bestellung in einen Bericht inkludiert wird." + +#: src/Settings.php:85 +msgctxt "oss" +msgid "Date paid" +msgstr "Zahlungsdatum" + +#: src/Settings.php:86 +msgctxt "oss" +msgid "Date created" +msgstr "Erstellungsdatum" + +#: src/Settings.php:97 +msgctxt "oss" +msgid "Fixed gross prices" +msgstr "Feste Bruttopreise" + +#: src/Settings.php:98 +msgctxt "oss" +msgid "Apply the same gross price regardless of the tax rate for EU countries." +msgstr "Verwende den Bruttopreis für EU-Länder unabhängig vom Steuersatz." + +#: src/Settings.php:98 +msgctxt "oss" +msgid "" +"This option will make sure that your customers pay the same price no matter " +"the tax rate (based on the country chosen) to be applied." +msgstr "" +"Diese Option bewirkt, dass Kunden, unabhängig vom Land und damit vom " +"Steuersatz den selben Bruttopreis bezahlen." + +#: src/Settings.php:104 +msgctxt "oss" +msgid "Third countries" +msgstr "Drittländer" + +#: src/Settings.php:105 +msgctxt "oss" +msgid "Apply the same gross price for third countries too." +msgstr "Verwende den Bruttopreis auch für Drittländer." + +#: src/Settings.php:138 src/Settings.php:139 +msgctxt "oss" +msgid "Are you sure? Please backup your tax rates before proceeding." +msgstr "Bist du sicher? Bitte erstelle vorab ein Backup deiner Steuersätze." + +#: src/Settings.php:138 +msgctxt "oss" +msgid "End OSS participation" +msgstr "OSS Teilnahme beenden" + +#: src/Settings.php:138 +msgctxt "oss" +msgid "Start OSS participation" +msgstr "OSS Teilnahme starten" + +#: src/Settings.php:139 +msgctxt "oss" +msgid "refresh VAT rates" +msgstr "Steuersätze erneuern" + +#: src/Settings.php:140 +msgctxt "oss" +msgid "learn more" +msgstr "Mehr erfahren" + +#: src/Settings.php:142 +msgctxt "oss" +msgid "" +"Use this option to automatically adjust tax-related options in WooCommerce. " +"Warning: This option will delete your current tax rates and add new tax " +"rates based on your OSS participation status." +msgstr "" +"Nutze diese Option um deine Steuereinstellungen in WooCommerce automatisch " +"anpassen zu lassen. Achtung: Diese Option löscht deine aktuellen Steuersätze " +"und fügt neue Steuersätze basierend auf deinem OSS Status hinzu." + +#: src/Settings.php:174 +msgctxt "oss" +msgid "See status" +msgstr "Status ansehen" + +#: src/Settings.php:174 +msgctxt "oss" +msgid "Start initial report" +msgstr "Initialen Bericht erstellen" + +#: src/Settings.php:175 +#, php-format +msgctxt "oss" +msgid "Report not yet completed. %s" +msgstr "Bericht noch nicht abgeschlossen. %s" + +#: src/Settings.php:175 +#, php-format +msgctxt "oss" +msgid "Report not yet started. %s" +msgstr "Bericht noch nicht gestartet. %s" + +#: src/Settings.php:193 +msgctxt "oss-amounts" +msgid "of" +msgstr "von" + +#: src/Settings.php:193 +#, php-format +msgctxt "oss" +msgid "As of: %s" +msgstr "Stand: %s" + +#: src/Settings.php:193 +msgctxt "oss" +msgid "see details" +msgstr "Details ansehen" + +#: src/Settings.php:194 +#, php-format +msgctxt "oss" +msgid "" +"This value indicates your current net total amount applicable for the One " +"Stop Shop procedure delivery threshold of the current year. You should take " +"action in case the delivery threshold is or is close to being exceeded. Find out more about the calculation." +msgstr "" +"Dieser Wert entspricht der Bemessungsgrundlage (Nettobetrag) des aktuellen " +"Jahres für die Lieferschwelle des One Stop Shop Verfahrens. Du solltest " +"tätig werden, wenn die Lieferschwelle kurz vor einer Überschreitung steht " +"oder bereits überschritten wurde. Erfahre mehr über die " +"Berechnung." + +#: src/SettingsPage.php:23 +msgctxt "oss" +msgid "Learn More" +msgstr "Mehr erfahren" + +#: src/Tax.php:260 src/Tax.php:338 +#, php-format +msgctxt "oss" +msgid "Tax class (%s)" +msgstr "Steuerklasse (%s)" + +#: src/Tax.php:261 +msgctxt "oss" +msgid "Same as parent" +msgstr "Gleiche wie übergeordnet" + +#: src/Tax.php:263 src/Tax.php:296 src/Tax.php:340 src/Tax.php:374 +msgctxt "oss" +msgid "remove" +msgstr "Löschen" + +#: src/Tax.php:273 src/Tax.php:351 +msgctxt "oss" +msgid "Add country specific tax class (OSS)" +msgstr "Länderspezifische Steuerklasse hinzufügen (OSS)" + +#: src/Tax.php:280 src/Tax.php:358 +msgctxt "oss" +msgid "Select country" +msgstr "Land auswählen" + +#: src/Tax.php:305 src/Tax.php:315 +msgctxt "oss" +msgid "EU-wide" +msgstr "EU-weit" + +#. translators: %s: Customer billing full name +#: templates/emails/admin-delivery-threshold.php:19 +#, php-format +msgctxt "oss" +msgid "" +"Your OSS delivery threshold of %1$s has been reached. Please take action " +"immediately. Visit the OSS Settings Panel for details." +msgstr "" +"Deine OSS Lieferschwelle von %1$s wurde erreicht. Bitte werde umgehend " +"tätig. Besuche die OSS Einstellungen um Details zu " +"erfahren." + +#: templates/emails/admin-delivery-threshold.php:21 +msgctxt "oss" +msgid "Report Details" +msgstr "Details des Berichts" + +#: templates/emails/admin-delivery-threshold.php:24 +msgctxt "oss" +msgid "Period" +msgstr "Periode" + +#: templates/emails/admin-delivery-threshold.php:29 +msgctxt "oss" +msgid "See report details" +msgstr "Details des Berichts abrufen" + +#: templates/emails/plain/admin-delivery-threshold.php:17 +#, php-format +msgctxt "oss" +msgid "" +"Your OSS delivery threshold of %1$s has been reached. Please take action " +"immediately. Visit the OSS Settings Panel (%2$s) for details." +msgstr "" +"Deine OSS Lieferschwelle von %1$s wurde erreicht. Bitte werde umgehend " +"tätig. Besuche die OSS Einstellungen (%2$s) um Details zu erfahren." + +#. Plugin Name of the plugin/theme +msgid "One Stop Shop for WooCommerce" +msgstr "One Stop Shop für WooCommerce" + +#. Plugin URI of the plugin/theme +msgid "https://github.com/vendidero/one-stop-shop-woocommerce" +msgstr "https://github.com/vendidero/one-stop-shop-woocommerce" + +#. Description of the plugin/theme +msgid "Comply with the One Stop Shop procedure while using WooCommerce." +msgstr "Nutze das One Stop Shop Verfahren zusammen mit WooCommerce." + +#. Author of the plugin/theme +msgid "vendidero" +msgstr "vendidero" + +#. Author URI of the plugin/theme +msgid "https://vendidero.de" +msgstr "https://vendidero.de" + +#~ msgctxt "storeabill-core" +#~ msgid "Year" +#~ msgstr "Jahr" + +#~ msgctxt "storeabill-core" +#~ msgid "Quarter" +#~ msgstr "Quartal" + +#~ msgctxt "storeabill-core" +#~ msgid "Month" +#~ msgstr "Monat" + +#~ msgctxt "storeabill-core" +#~ msgid "Date range" +#~ msgstr "Zeitraum" + +#~ msgctxt "oss" +#~ msgid "" +#~ "Find useful options regarding the One Stop Shop procedure here." +#~ msgstr "" +#~ "Finde hier nützliche Optionen zum One Stop Shop Verfahren." + +#, php-format +#~ msgctxt "oss-tax-rate-import" +#~ msgid "VAT %s" +#~ msgstr "MwSt. %s" diff --git a/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce-de_DE_formal.mo b/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce-de_DE_formal.mo new file mode 100644 index 000000000..11a667eef Binary files /dev/null and b/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce-de_DE_formal.mo differ diff --git a/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce-de_DE_formal.po b/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce-de_DE_formal.po new file mode 100644 index 000000000..87a7842c5 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce-de_DE_formal.po @@ -0,0 +1,788 @@ +msgid "" +msgstr "" +"Project-Id-Version: One Stop Shop for WooCommerce\n" +"POT-Creation-Date: 2022-12-09 11:36+0100\n" +"PO-Revision-Date: 2022-12-09 11:36+0100\n" +"Last-Translator: Dennis Nissle \n" +"Language-Team: \n" +"Language: de_DE@formal\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Generator: Poedit 3.1.1\n" +"X-Poedit-Basepath: ../..\n" +"X-Poedit-Flags-xgettext: --add-comments=translators:\n" +"X-Poedit-WPHeader: one-stop-shop-woocommerce.php\n" +"X-Poedit-SourceCharset: UTF-8\n" +"X-Poedit-KeywordsList: __;_e;_n:1,2;_x:1,2c;_ex:1,2c;_nx:4c,1,2;esc_attr__;" +"esc_attr_e;esc_attr_x:1,2c;esc_html__;esc_html_e;esc_html_x:1,2c;_n_noop:1,2;" +"_nx_noop:3c,1,2;__ngettext_noop:1,2\n" +"X-Poedit-SearchPath-0: .\n" +"X-Poedit-SearchPathExcluded-0: *.min.js\n" +"X-Poedit-SearchPathExcluded-1: vendor\n" +"X-Poedit-SearchPathExcluded-2: node_modules\n" +"X-Poedit-SearchPathExcluded-3: build\n" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:312 +msgid "Reduced rate" +msgstr "" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:313 +msgctxt "tax-helper-tax-class-name" +msgid "Greater reduced rate" +msgstr "Zusätzlicher reduzierter Preis" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:314 +msgctxt "tax-helper-tax-class-name" +msgid "Super reduced rate" +msgstr "Stark reduzierter Preis" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:718 +msgctxt "tax-helper" +msgid "Madeira" +msgstr "Madeira" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:725 +msgctxt "tax-helper" +msgid "Acores" +msgstr "Azoren" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:761 +msgctxt "tax-helper" +msgid "Northern Ireland" +msgstr "Nordirland" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:774 +msgctxt "tax-helper-rate-import" +msgid "Exempt" +msgstr "Ausnahme" + +#: libs/woocommerce-eu-tax-helper/src/Helper.php:843 +#, php-format +msgctxt "tax-helper-rate-import" +msgid "VAT %1$s %% %2$s" +msgstr "MwSt. %1$s %% %2$s" + +#. translators: 1: composer command. 2: plugin directory +#: one-stop-shop-woocommerce.php:44 one-stop-shop-woocommerce.php:62 +#, php-format +msgctxt "oss" +msgid "" +"Your installation of the One Stop Shop feature plugin is incomplete. Please " +"run %1$s within the %2$s directory." +msgstr "" +"Ihre Installation des One Stop Shop Feature Plugins ist nicht komplett. " +"Bitte führen Sie %1$s innerhalb des %2$s Verzeichnisses aus." + +#: src/Admin.php:78 src/Admin.php:79 +msgctxt "oss" +msgid "Refresh VAT rates (OSS)" +msgstr "EU Steuersätze erneuern (OSS)" + +#: src/Admin.php:83 +msgctxt "oss" +msgid "Note:" +msgstr "Hinweis:" + +#: src/Admin.php:84 +#, php-format +msgctxt "oss" +msgid "" +"This option will delete all of your current EU VAT rates and re-import them " +"based on your current OSS status." +msgstr "" +"Diese Option löscht alle Ihre aktuellen EU Steuersätze und importiert sie " +"anschließend auf Basis Ihres OSS Status neu." + +#: src/Admin.php:214 +#, php-format +msgctxt "oss" +msgid "" +"Seems like you have reached (or are close to reaching) the delivery " +"threshold for the current year. Please make sure to check the report details and take action in case necessary." +msgstr "" +"Sie sind kurz davor oder haben bereits die Lieferschwelle für das aktuelle " +"Jahr überschritten. Bitte prüfen Sie die Details des Berichts und unternehmen Sie weitere Schritte." + +#: src/Admin.php:218 +msgctxt "oss" +msgid "Delivery threshold reached (OSS)" +msgstr "Lieferschwelle erreicht (OSS)" + +#: src/Admin.php:448 src/SettingsPage.php:17 +msgctxt "oss" +msgid "OSS" +msgstr "OSS" + +#: src/Admin.php:448 src/Admin.php:566 src/SettingsPage.php:23 +msgctxt "oss" +msgid "One Stop Shop" +msgstr "One Stop Shop" + +#: src/Admin.php:469 src/Package.php:365 +#, php-format +msgctxt "oss" +msgid "Q%1$s/%2$s" +msgstr "Q%1$s/%2$s" + +#: src/Admin.php:478 src/Package.php:370 +#, php-format +msgctxt "oss" +msgid "%1$s/%2$s" +msgstr "%1$s/%2$s" + +#: src/Admin.php:486 +msgctxt "oss" +msgid "New Report" +msgstr "Neuer Bericht" + +#: src/Admin.php:493 +msgctxt "oss" +msgid "Type" +msgstr "Typ" + +#: src/Admin.php:505 +msgctxt "oss" +msgid "Year" +msgstr "Jahr" + +#: src/Admin.php:517 +msgctxt "oss" +msgid "Quarter" +msgstr "Quartal" + +#: src/Admin.php:529 +msgctxt "oss" +msgid "Month" +msgstr "Monat" + +#: src/Admin.php:541 +msgctxt "oss" +msgid "Date range" +msgstr "Zeitraum" + +#: src/Admin.php:553 +msgctxt "oss" +msgid "Start report" +msgstr "Bericht starten" + +#: src/Admin.php:567 +msgctxt "oss" +msgid "New report" +msgstr "Neuer Bericht" + +#: src/Admin.php:609 +msgctxt "oss" +msgid "View" +msgstr "Ansehen" + +#: src/Admin.php:613 +msgctxt "oss" +msgid "Export" +msgstr "Exportieren" + +#: src/Admin.php:617 +msgctxt "oss" +msgid "Export BOP" +msgstr "BOP-CSV exportieren" + +#: src/Admin.php:621 +msgctxt "oss" +msgid "Refresh" +msgstr "Aktualisieren" + +#: src/Admin.php:625 +msgctxt "oss" +msgid "Delete" +msgstr "Löschen" + +#: src/Admin.php:631 +msgctxt "oss" +msgid "Cancel" +msgstr "Abbrechen" + +#: src/Admin.php:669 +msgctxt "oss" +msgid "Country" +msgstr "Land" + +#: src/Admin.php:670 +msgctxt "oss" +msgid "Tax Rate" +msgstr "Steuersatz" + +#: src/Admin.php:671 +msgctxt "oss" +msgid "Net Total" +msgstr "Nettobetrag" + +#: src/Admin.php:672 +msgctxt "oss" +msgid "Tax Total" +msgstr "Steuerbertrag" + +#: src/Admin.php:703 +#, php-format +msgctxt "oss" +msgid "%1$s %%" +msgstr "%1$s %%" + +#: src/Admin.php:716 +#, php-format +msgctxt "oss" +msgid "" +"Currently processed %1$s orders. Next iteration is scheduled for %2$s. Find pending actions" +msgstr "" +"Aktuell %1$s Bestellung verarbeitet. Nächster Durchlauf planmäßig am %2$s. " +"Offene Aktionen finden" + +#: src/Admin.php:716 +msgctxt "oss" +msgid "Not yet known" +msgstr "Noch nicht bekannt" + +#: src/Admin.php:747 src/ReportTable.php:39 src/SettingsPage.php:23 +msgctxt "oss" +msgid "Reports" +msgstr "Berichte" + +#: src/AdminNote.php:46 +msgctxt "oss" +msgid "Dismiss" +msgstr "Ausblenden" + +#: src/AsyncReportGenerator.php:258 +msgctxt "oss" +msgid "No orders found." +msgstr "Keine Bestellungen gefunden." + +#: src/CSVExporter.php:54 +msgctxt "oss" +msgid "Country code" +msgstr "Land des Verbrauchs" + +#: src/CSVExporter.php:55 +msgctxt "oss" +msgid "Tax rate" +msgstr "Umsatzsteuersatz" + +#: src/CSVExporter.php:56 +msgctxt "oss" +msgid "Taxable base" +msgstr "Nettobetrag" + +#: src/CSVExporter.php:57 +msgctxt "oss" +msgid "Amount" +msgstr "Umsatzsteuerbetrag" + +#: src/DeliveryThresholdEmailNotification.php:19 +msgctxt "oss" +msgid "OSS Delivery Threshold Notification" +msgstr "OSS Lieferschwelle Benachrichtigung" + +#: src/DeliveryThresholdEmailNotification.php:20 +msgctxt "oss" +msgid "" +"This email notifies shop owners in case the delivery threshold (OSS) is " +"close to being reached." +msgstr "" +"Diese E-Mail benachrichtigt den Shopbetreiber über eine in Kürze anstehende " +"Überschreitung der Lieferschwelle (OSS)." + +#: src/DeliveryThresholdEmailNotification.php:38 +msgctxt "oss" +msgid "[{site_title}]: OSS delivery threshold reached" +msgstr "[{site_title}]: OSS Lieferschwelle erreicht" + +#: src/DeliveryThresholdEmailNotification.php:48 +msgctxt "oss" +msgid "OSS delivery threshold reached" +msgstr "OSS Lieferschwelle erreicht" + +#: src/DeliveryThresholdWarning.php:14 +msgctxt "oss" +msgid "See details" +msgstr "Details ansehen" + +#: src/Package.php:178 +msgctxt "oss" +msgid "" +"To use the OSS for WooCommerce plugin please make sure that WooCommerce is " +"installed and activated." +msgstr "" +"Um das OSS für WooCommerce Plugin nutzen zu können muss WooCommerce " +"installiert und aktiviert sein." + +#: src/Package.php:350 src/ReportTable.php:40 +msgctxt "oss" +msgid "Report" +msgstr "Bericht" + +#: src/Package.php:374 +#, php-format +msgctxt "oss" +msgid "%1$s" +msgstr "%1$s" + +#: src/Package.php:379 +#, php-format +msgctxt "oss" +msgid "%1$s - %2$s" +msgstr "%1$s - %2$s" + +#: src/Package.php:384 +#, php-format +msgctxt "oss" +msgid "Observer %1$s" +msgstr "Beobachter %1$s" + +#: src/Package.php:636 +msgctxt "oss" +msgid "Quarterly" +msgstr "Quartalsweise" + +#: src/Package.php:637 +msgctxt "oss" +msgid "Yearly" +msgstr "Jährlich" + +#: src/Package.php:638 +msgctxt "oss" +msgid "Monthly" +msgstr "Monatlich" + +#: src/Package.php:639 +msgctxt "oss" +msgid "Custom" +msgstr "Individuell" + +#: src/Package.php:643 +msgctxt "oss" +msgid "Observer" +msgstr "Beobachter" + +#: src/Package.php:657 +msgctxt "oss" +msgid "Pending" +msgstr "In Bearbeitung" + +#: src/Package.php:658 +msgctxt "oss" +msgid "Completed" +msgstr "Fertiggestellt" + +#: src/Package.php:659 +msgctxt "oss" +msgid "Failed" +msgstr "Fehlgeschlagen" + +#: src/ReportTable.php:124 +#, php-format +msgctxt "oss" +msgid "%d report deleted." +msgid_plural "%d reports deleted." +msgstr[0] "%d Bericht gelöscht." +msgstr[1] "%d Berichte gelöscht." + +#: src/ReportTable.php:187 +msgctxt "oss" +msgid "No reports found" +msgstr "Keine Berichte gefunden" + +#: src/ReportTable.php:228 +#, php-format +msgctxt "oss" +msgid "All (%s)" +msgid_plural "All (%s)" +msgstr[0] "Alle (%s)" +msgstr[1] "Alle (%s)" + +#: src/ReportTable.php:255 +#, php-format +msgctxt "oss" +msgid " (%s)" +msgid_plural " (%s)" +msgstr[0] " (%s)" +msgstr[1] " (%s)" + +#: src/ReportTable.php:326 +msgctxt "oss" +msgid "Filter" +msgstr "Filtern" + +#: src/ReportTable.php:353 +msgctxt "oss" +msgid "Title" +msgstr "Titel" + +#: src/ReportTable.php:354 +msgctxt "oss" +msgid "Start" +msgstr "Start" + +#: src/ReportTable.php:355 +msgctxt "oss" +msgid "End" +msgstr "Ende" + +#: src/ReportTable.php:356 templates/emails/admin-delivery-threshold.php:25 +msgctxt "oss" +msgid "Net total" +msgstr "Nettobetrag" + +#: src/ReportTable.php:357 templates/emails/admin-delivery-threshold.php:26 +msgctxt "oss" +msgid "Tax total" +msgstr "Steuerbertrag" + +#: src/ReportTable.php:358 +msgctxt "oss" +msgid "Status" +msgstr "Status" + +#: src/ReportTable.php:359 +msgctxt "oss" +msgid "Actions" +msgstr "Aktionen" + +#: src/ReportTable.php:423 +#, php-format +msgctxt "oss" +msgid "Select %s" +msgstr "%s auswählen" + +#: src/ReportTable.php:516 +msgctxt "oss" +msgid "Delete Permanently" +msgstr "Unwiderruflich löschen" + +#: src/Settings.php:16 +msgctxt "oss" +msgid "General" +msgstr "Allgemein" + +#: src/Settings.php:21 +msgctxt "oss" +msgid "" +"Find useful options regarding the One Stop Shop procedure here." +msgstr "" +"Finden Sie hier nützliche Optionen zum One Stop Shop Verfahren." + +#: src/Settings.php:38 +msgctxt "oss" +msgid "OSS status" +msgstr "OSS Status" + +#: src/Settings.php:39 +msgctxt "oss" +msgid "Yes, I'm currently participating in the OSS procedure." +msgstr "Ja, ich nehme aktuell am One Stop Shop Verfahren teil." + +#: src/Settings.php:46 +msgctxt "oss" +msgid "Observation" +msgstr "Überwachung" + +#: src/Settings.php:47 +msgctxt "oss" +msgid "Automatically observe the delivery threshold of the current year." +msgstr "Überwache die Lieferschwelle des aktuellen Jahres automatisch." + +#: src/Settings.php:47 +msgctxt "oss" +msgid "" +"This option will automatically calculate the amount applicable for the OSS " +"procedure delivery threshold once per day for the current year. The report " +"will only recalculated for the days which are not yet subject to the " +"observation to save processing time." +msgstr "" +"Diese Option überwacht automatisch die Lieferschwelle des OSS Verfahrens für " +"das aktuelle Jahr, indem der dazugehörige Bericht täglich erweitert und " +"nachberechnet wird. Der Bericht wird nur für die Tage nachberechnet, für die " +"aktuell noch keine Überwachung stattfindet." + +#: src/Settings.php:59 +msgctxt "oss" +msgid "Delivery threshold" +msgstr "Lieferschwelle" + +#: src/Settings.php:72 +msgctxt "oss" +msgid "Participation" +msgstr "Teilnahme" + +#: src/Settings.php:79 +msgctxt "oss" +msgid "Report Order Date" +msgstr "Bericht Bestelldatum" + +#: src/Settings.php:80 +msgctxt "oss" +msgid "" +"Select the relevant order date to be used to determine whether to include an " +"order in a report." +msgstr "" +"Wählen Sie das relevante Bestelldatum aus auf Basis dessen entschieden wird " +"ob eine Bestellung in einen Bericht inkludiert wird." + +#: src/Settings.php:85 +msgctxt "oss" +msgid "Date paid" +msgstr "Zahlungsdatum" + +#: src/Settings.php:86 +msgctxt "oss" +msgid "Date created" +msgstr "Erstellungsdatum" + +#: src/Settings.php:97 +msgctxt "oss" +msgid "Fixed gross prices" +msgstr "Feste Bruttopreise" + +#: src/Settings.php:98 +msgctxt "oss" +msgid "Apply the same gross price regardless of the tax rate for EU countries." +msgstr "Verwende den Bruttopreis für EU-Länder unabhängig vom Steuersatz." + +#: src/Settings.php:98 +msgctxt "oss" +msgid "" +"This option will make sure that your customers pay the same price no matter " +"the tax rate (based on the country chosen) to be applied." +msgstr "" +"Diese Option bewirkt, dass Kunden, unabhängig vom Land und damit vom " +"Steuersatz den selben Bruttopreis bezahlen." + +#: src/Settings.php:104 +msgctxt "oss" +msgid "Third countries" +msgstr "Drittländer" + +#: src/Settings.php:105 +msgctxt "oss" +msgid "Apply the same gross price for third countries too." +msgstr "Verwende den Bruttopreis auch für Drittländer." + +#: src/Settings.php:138 src/Settings.php:139 +msgctxt "oss" +msgid "Are you sure? Please backup your tax rates before proceeding." +msgstr "" +"Sind Sie sicher? Bitte erstellen Sie vorab ein Backup Ihrer Steuersätze." + +#: src/Settings.php:138 +msgctxt "oss" +msgid "End OSS participation" +msgstr "OSS Teilnahme beenden" + +#: src/Settings.php:138 +msgctxt "oss" +msgid "Start OSS participation" +msgstr "OSS Teilnahme starten" + +#: src/Settings.php:139 +msgctxt "oss" +msgid "refresh VAT rates" +msgstr "Steuersätze erneuern" + +#: src/Settings.php:140 +msgctxt "oss" +msgid "learn more" +msgstr "Mehr erfahren" + +#: src/Settings.php:142 +msgctxt "oss" +msgid "" +"Use this option to automatically adjust tax-related options in WooCommerce. " +"Warning: This option will delete your current tax rates and add new tax " +"rates based on your OSS participation status." +msgstr "" +"Nutzen Sie diese Option um Ihre Steuereinstellungen in WooCommerce " +"automatisch anpassen zu lassen. Achtung: Diese Option löscht Ihre aktuellen " +"Steuersätze und fügt neue Steuersätze basierend auf Ihrem OSS Status hinzu." + +#: src/Settings.php:174 +msgctxt "oss" +msgid "See status" +msgstr "Status ansehen" + +#: src/Settings.php:174 +msgctxt "oss" +msgid "Start initial report" +msgstr "Initialen Bericht erstellen" + +#: src/Settings.php:175 +#, php-format +msgctxt "oss" +msgid "Report not yet completed. %s" +msgstr "Bericht noch nicht abgeschlossen. %s" + +#: src/Settings.php:175 +#, php-format +msgctxt "oss" +msgid "Report not yet started. %s" +msgstr "Bericht noch nicht gestartet. %s" + +#: src/Settings.php:193 +msgctxt "oss-amounts" +msgid "of" +msgstr "von" + +#: src/Settings.php:193 +#, php-format +msgctxt "oss" +msgid "As of: %s" +msgstr "Stand: %s" + +#: src/Settings.php:193 +msgctxt "oss" +msgid "see details" +msgstr "Details ansehen" + +#: src/Settings.php:194 +#, php-format +msgctxt "oss" +msgid "" +"This value indicates your current net total amount applicable for the One " +"Stop Shop procedure delivery threshold of the current year. You should take " +"action in case the delivery threshold is or is close to being exceeded. Find out more about the calculation." +msgstr "" +"Dieser Wert entspricht der Bemessungsgrundlage (Nettobetrag) des aktuellen " +"Jahres für die Lieferschwelle des One Stop Shop Verfahrens. Sie sollten " +"tätig werden, wenn die Lieferschwelle kurz vor einer Überschreitung steht " +"oder bereits überschritten wurde. Erfahren Sie mehr über " +"die Berechnung." + +#: src/SettingsPage.php:23 +msgctxt "oss" +msgid "Learn More" +msgstr "Mehr erfahren" + +#: src/Tax.php:260 src/Tax.php:338 +#, php-format +msgctxt "oss" +msgid "Tax class (%s)" +msgstr "Steuerklasse (%s)" + +#: src/Tax.php:261 +msgctxt "oss" +msgid "Same as parent" +msgstr "Gleiche wie übergeordnet" + +#: src/Tax.php:263 src/Tax.php:296 src/Tax.php:340 src/Tax.php:374 +msgctxt "oss" +msgid "remove" +msgstr "Löschen" + +#: src/Tax.php:273 src/Tax.php:351 +msgctxt "oss" +msgid "Add country specific tax class (OSS)" +msgstr "Länderspezifische Steuerklasse hinzufügen (OSS)" + +#: src/Tax.php:280 src/Tax.php:358 +msgctxt "oss" +msgid "Select country" +msgstr "Land auswählen" + +#: src/Tax.php:305 src/Tax.php:315 +msgctxt "oss" +msgid "EU-wide" +msgstr "EU-weit" + +#. translators: %s: Customer billing full name +#: templates/emails/admin-delivery-threshold.php:19 +#, php-format +msgctxt "oss" +msgid "" +"Your OSS delivery threshold of %1$s has been reached. Please take action " +"immediately. Visit the OSS Settings Panel for details." +msgstr "" +"Ihre OSS Lieferschwelle von %1$s wurde erreicht. Bitte werden Sie umgehend " +"tätig. Besuchen Sie die OSS Einstellungen um Details zu " +"erfahren." + +#: templates/emails/admin-delivery-threshold.php:21 +msgctxt "oss" +msgid "Report Details" +msgstr "Details des Berichts" + +#: templates/emails/admin-delivery-threshold.php:24 +msgctxt "oss" +msgid "Period" +msgstr "Periode" + +#: templates/emails/admin-delivery-threshold.php:29 +msgctxt "oss" +msgid "See report details" +msgstr "Details des Berichts abrufen" + +#: templates/emails/plain/admin-delivery-threshold.php:17 +#, php-format +msgctxt "oss" +msgid "" +"Your OSS delivery threshold of %1$s has been reached. Please take action " +"immediately. Visit the OSS Settings Panel (%2$s) for details." +msgstr "" +"Ihre OSS Lieferschwelle von %1$s wurde erreicht. Bitte werden Sie umgehend " +"tätig. Besuchen Sie die OSS Einstellungen (%2$s) um Details zu erfahren." + +#. Plugin Name of the plugin/theme +msgid "One Stop Shop for WooCommerce" +msgstr "One Stop Shop für WooCommerce" + +#. Plugin URI of the plugin/theme +msgid "https://github.com/vendidero/one-stop-shop-woocommerce" +msgstr "https://github.com/vendidero/one-stop-shop-woocommerce" + +#. Description of the plugin/theme +msgid "Comply with the One Stop Shop procedure while using WooCommerce." +msgstr "Nutzen Sie das One Stop Shop Verfahren zusammen mit WooCommerce." + +#. Author of the plugin/theme +msgid "vendidero" +msgstr "vendidero" + +#. Author URI of the plugin/theme +msgid "https://vendidero.de" +msgstr "https://vendidero.de" + +#~ msgctxt "storeabill-core" +#~ msgid "Year" +#~ msgstr "Jahr" + +#~ msgctxt "storeabill-core" +#~ msgid "Quarter" +#~ msgstr "Quartal" + +#~ msgctxt "storeabill-core" +#~ msgid "Month" +#~ msgstr "Monat" + +#~ msgctxt "storeabill-core" +#~ msgid "Date range" +#~ msgstr "Zeitraum" + +#~ msgctxt "oss" +#~ msgid "" +#~ "Find useful options regarding the One Stop Shop procedure here." +#~ msgstr "" +#~ "Finden Sie hier nützliche Optionen zum One Stop Shop Verfahren." + +#, php-format +#~ msgctxt "oss-tax-rate-import" +#~ msgid "VAT %s" +#~ msgstr "MwSt. %s" diff --git a/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce.pot b/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce.pot new file mode 100644 index 000000000..bc7292f91 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/i18n/languages/one-stop-shop-woocommerce.pot @@ -0,0 +1,673 @@ +#, fuzzy +msgid "" +msgstr "" +"Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n" +"Project-Id-Version: One Stop Shop for WooCommerce\n" +"POT-Creation-Date: 2021-07-08 13:56+0200\n" +"PO-Revision-Date: 2021-05-10 09:48+0200\n" +"Last-Translator: \n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"X-Generator: Poedit 2.4.3\n" +"X-Poedit-Basepath: ../..\n" +"X-Poedit-Flags-xgettext: --add-comments=translators:\n" +"X-Poedit-WPHeader: one-stop-shop-woocommerce.php\n" +"X-Poedit-SourceCharset: UTF-8\n" +"X-Poedit-KeywordsList: __;_e;_n:1,2;_x:1,2c;_ex:1,2c;_nx:4c,1,2;esc_attr__;" +"esc_attr_e;esc_attr_x:1,2c;esc_html__;esc_html_e;esc_html_x:1,2c;_n_noop:1,2;" +"_nx_noop:3c,1,2;__ngettext_noop:1,2\n" +"X-Poedit-SearchPath-0: .\n" +"X-Poedit-SearchPathExcluded-0: *.min.js\n" +"X-Poedit-SearchPathExcluded-1: vendor\n" +"X-Poedit-SearchPathExcluded-2: node_modules\n" +"X-Poedit-SearchPathExcluded-3: build\n" + +#. translators: 1: composer command. 2: plugin directory +#: one-stop-shop-woocommerce.php:44 one-stop-shop-woocommerce.php:62 +#, php-format +msgctxt "oss" +msgid "" +"Your installation of the One Stop Shop feature plugin is incomplete. Please " +"run %1$s within the %2$s directory." +msgstr "" + +#: src/Admin.php:76 src/Admin.php:77 +msgctxt "oss" +msgid "Refresh VAT rates (OSS)" +msgstr "" + +#: src/Admin.php:81 +msgctxt "oss" +msgid "Note:" +msgstr "" + +#: src/Admin.php:82 +#, php-format +msgctxt "oss" +msgid "" +"This option will delete all of your current EU VAT rates and re-import them " +"based on your current OSS status." +msgstr "" + +#: src/Admin.php:212 +#, php-format +msgctxt "oss" +msgid "" +"Seems like you have reached (or are close to reaching) the delivery " +"threshold for the current year. Please make sure to check the report details and take action in case necessary." +msgstr "" + +#: src/Admin.php:216 +msgctxt "oss" +msgid "Delivery threshold reached (OSS)" +msgstr "" + +#: src/Admin.php:432 src/SettingsPage.php:17 +msgctxt "oss" +msgid "OSS" +msgstr "" + +#: src/Admin.php:432 src/Admin.php:550 src/SettingsPage.php:23 +msgctxt "oss" +msgid "One Stop Shop" +msgstr "" + +#: src/Admin.php:453 src/Package.php:264 +#, php-format +msgctxt "oss" +msgid "Q%1$s/%2$s" +msgstr "" + +#: src/Admin.php:462 src/Package.php:269 +#, php-format +msgctxt "oss" +msgid "%1$s/%2$s" +msgstr "" + +#: src/Admin.php:470 +msgctxt "oss" +msgid "New Report" +msgstr "" + +#: src/Admin.php:477 +msgctxt "oss" +msgid "Type" +msgstr "" + +#: src/Admin.php:489 +msgctxt "oss" +msgid "Year" +msgstr "" + +#: src/Admin.php:501 +msgctxt "oss" +msgid "Quarter" +msgstr "" + +#: src/Admin.php:513 +msgctxt "oss" +msgid "Month" +msgstr "" + +#: src/Admin.php:525 +msgctxt "oss" +msgid "Date range" +msgstr "" + +#: src/Admin.php:537 +msgctxt "oss" +msgid "Start report" +msgstr "" + +#: src/Admin.php:551 +msgctxt "oss" +msgid "New report" +msgstr "" + +#: src/Admin.php:593 +msgctxt "oss" +msgid "View" +msgstr "" + +#: src/Admin.php:597 +msgctxt "oss" +msgid "Export" +msgstr "" + +#: src/Admin.php:601 +msgctxt "oss" +msgid "Refresh" +msgstr "" + +#: src/Admin.php:605 +msgctxt "oss" +msgid "Delete" +msgstr "" + +#: src/Admin.php:611 +msgctxt "oss" +msgid "Cancel" +msgstr "" + +#: src/Admin.php:640 +msgctxt "oss" +msgid "Country" +msgstr "" + +#: src/Admin.php:641 +msgctxt "oss" +msgid "Tax Rate" +msgstr "" + +#: src/Admin.php:642 +msgctxt "oss" +msgid "Net Total" +msgstr "" + +#: src/Admin.php:643 +msgctxt "oss" +msgid "Tax Total" +msgstr "" + +#: src/Admin.php:674 +#, php-format +msgctxt "oss" +msgid "%1$s %%" +msgstr "" + +#: src/Admin.php:686 +#, php-format +msgctxt "oss" +msgid "" +"Currently processed %1$s orders. Next iteration is scheduled for %2$s. Find pending actions" +msgstr "" + +#: src/Admin.php:686 +msgctxt "oss" +msgid "Not yet known" +msgstr "" + +#: src/Admin.php:717 src/ReportTable.php:39 src/SettingsPage.php:23 +msgctxt "oss" +msgid "Reports" +msgstr "" + +#: src/AdminNote.php:39 +msgctxt "oss" +msgid "Dismiss" +msgstr "" + +#: src/AsyncReportGenerator.php:244 +msgctxt "oss" +msgid "No orders found." +msgstr "" + +#: src/CSVExporter.php:52 +msgctxt "oss" +msgid "Country code" +msgstr "" + +#: src/CSVExporter.php:53 +msgctxt "oss" +msgid "Tax rate" +msgstr "" + +#: src/CSVExporter.php:54 +msgctxt "oss" +msgid "Taxable base" +msgstr "" + +#: src/CSVExporter.php:55 +msgctxt "oss" +msgid "Amount" +msgstr "" + +#: src/DeliveryThresholdEmailNotification.php:19 +msgctxt "oss" +msgid "OSS Delivery Threshold Notification" +msgstr "" + +#: src/DeliveryThresholdEmailNotification.php:20 +msgctxt "oss" +msgid "" +"This email notifies shop owners in case the delivery threshold (OSS) is " +"close to being reached." +msgstr "" + +#: src/DeliveryThresholdEmailNotification.php:38 +msgctxt "oss" +msgid "[{site_title}]: OSS delivery threshold reached" +msgstr "" + +#: src/DeliveryThresholdEmailNotification.php:48 +msgctxt "oss" +msgid "OSS delivery threshold reached" +msgstr "" + +#: src/DeliveryThresholdWarning.php:13 +msgctxt "oss" +msgid "See details" +msgstr "" + +#: src/Package.php:80 +msgctxt "oss" +msgid "" +"To use the OSS for WooCommerce plugin please make sure that WooCommerce is " +"installed and activated." +msgstr "" + +#: src/Package.php:249 src/ReportTable.php:40 +msgctxt "oss" +msgid "Report" +msgstr "" + +#: src/Package.php:273 +#, php-format +msgctxt "oss" +msgid "%1$s" +msgstr "" + +#: src/Package.php:278 +#, php-format +msgctxt "oss" +msgid "%1$s - %2$s" +msgstr "" + +#: src/Package.php:283 +#, php-format +msgctxt "oss" +msgid "Observer %1$s" +msgstr "" + +#: src/Package.php:482 +msgctxt "oss" +msgid "Quarterly" +msgstr "" + +#: src/Package.php:483 +msgctxt "oss" +msgid "Yearly" +msgstr "" + +#: src/Package.php:484 +msgctxt "oss" +msgid "Monthly" +msgstr "" + +#: src/Package.php:485 +msgctxt "oss" +msgid "Custom" +msgstr "" + +#: src/Package.php:489 +msgctxt "oss" +msgid "Observer" +msgstr "" + +#: src/Package.php:503 +msgctxt "oss" +msgid "Pending" +msgstr "" + +#: src/Package.php:504 +msgctxt "oss" +msgid "Completed" +msgstr "" + +#: src/Package.php:505 +msgctxt "oss" +msgid "Failed" +msgstr "" + +#: src/ReportTable.php:121 +#, php-format +msgctxt "oss" +msgid "%d report deleted." +msgid_plural "%d reports deleted." +msgstr[0] "" +msgstr[1] "" + +#: src/ReportTable.php:184 +msgctxt "oss" +msgid "No reports found" +msgstr "" + +#: src/ReportTable.php:225 +#, php-format +msgctxt "oss" +msgid "All (%s)" +msgid_plural "All (%s)" +msgstr[0] "" +msgstr[1] "" + +#: src/ReportTable.php:250 +#, php-format +msgctxt "oss" +msgid " (%s)" +msgid_plural " (%s)" +msgstr[0] "" +msgstr[1] "" + +#: src/ReportTable.php:323 +msgctxt "oss" +msgid "Filter" +msgstr "" + +#: src/ReportTable.php:350 +msgctxt "oss" +msgid "Title" +msgstr "" + +#: src/ReportTable.php:351 +msgctxt "oss" +msgid "Start" +msgstr "" + +#: src/ReportTable.php:352 +msgctxt "oss" +msgid "End" +msgstr "" + +#: src/ReportTable.php:353 templates/emails/admin-delivery-threshold.php:25 +msgctxt "oss" +msgid "Net total" +msgstr "" + +#: src/ReportTable.php:354 templates/emails/admin-delivery-threshold.php:26 +msgctxt "oss" +msgid "Tax total" +msgstr "" + +#: src/ReportTable.php:355 +msgctxt "oss" +msgid "Status" +msgstr "" + +#: src/ReportTable.php:356 +msgctxt "oss" +msgid "Actions" +msgstr "" + +#: src/ReportTable.php:420 +#, php-format +msgctxt "oss" +msgid "Select %s" +msgstr "" + +#: src/ReportTable.php:513 +msgctxt "oss" +msgid "Delete Permanently" +msgstr "" + +#: src/Settings.php:14 +msgctxt "oss" +msgid "General" +msgstr "" + +#: src/Settings.php:19 +msgctxt "oss" +msgid "" +"Find useful options regarding the One Stop Shop procedure here." +msgstr "" + +#: src/Settings.php:31 +msgctxt "oss" +msgid "OSS status" +msgstr "" + +#: src/Settings.php:32 +msgctxt "oss" +msgid "Yes, I'm currently participating in the OSS procedure." +msgstr "" + +#: src/Settings.php:39 +msgctxt "oss" +msgid "Observation" +msgstr "" + +#: src/Settings.php:40 +msgctxt "oss" +msgid "Automatically observe the delivery threshold of the current year." +msgstr "" + +#: src/Settings.php:40 +msgctxt "oss" +msgid "" +"This option will automatically calculate the amount applicable for the OSS " +"procedure delivery threshold once per day for the current year. The report " +"will only recalculated for the days which are not yet subject to the " +"observation to save processing time." +msgstr "" + +#: src/Settings.php:50 +msgctxt "oss" +msgid "Delivery threshold" +msgstr "" + +#: src/Settings.php:60 +msgctxt "oss" +msgid "Participation" +msgstr "" + +#: src/Settings.php:70 +msgctxt "oss" +msgid "Fixed gross prices" +msgstr "" + +#: src/Settings.php:71 +msgctxt "oss" +msgid "Apply the same gross price regardless of the tax rate." +msgstr "" + +#: src/Settings.php:71 +msgctxt "oss" +msgid "" +"This option will make sure that your customers pay the same price no matter " +"the tax rate (based on the country chosen) to be applied." +msgstr "" + +#: src/Settings.php:92 +msgctxt "oss" +msgid "Are you sure? Please backup your tax rates before proceeding." +msgstr "" + +#: src/Settings.php:92 +msgctxt "oss" +msgid "End OSS participation" +msgstr "" + +#: src/Settings.php:92 +msgctxt "oss" +msgid "Start OSS participation" +msgstr "" + +#: src/Settings.php:93 +msgctxt "oss" +msgid "learn more" +msgstr "" + +#: src/Settings.php:95 +msgctxt "oss" +msgid "" +"Use this option to automatically adjust tax-related options in WooCommerce. " +"Warning: This option will delete your current tax rates and add new tax " +"rates based on your OSS participation status." +msgstr "" + +#: src/Settings.php:127 +msgctxt "oss" +msgid "See status" +msgstr "" + +#: src/Settings.php:127 +msgctxt "oss" +msgid "Start initial report" +msgstr "" + +#: src/Settings.php:128 +#, php-format +msgctxt "oss" +msgid "Report not yet completed. %s" +msgstr "" + +#: src/Settings.php:128 +#, php-format +msgctxt "oss" +msgid "Report not yet started. %s" +msgstr "" + +#: src/Settings.php:146 +msgctxt "oss-amounts" +msgid "of" +msgstr "" + +#: src/Settings.php:146 +#, php-format +msgctxt "oss" +msgid "As of: %s" +msgstr "" + +#: src/Settings.php:146 +msgctxt "oss" +msgid "see details" +msgstr "" + +#: src/Settings.php:147 +#, php-format +msgctxt "oss" +msgid "" +"This value indicates your current net total amount applicable for the One " +"Stop Shop procedure delivery threshold of the current year. You should take " +"action in case the delivery threshold is or is close to being exceeded. Find out more about the calculation." +msgstr "" + +#: src/SettingsPage.php:23 +msgctxt "oss" +msgid "Learn More" +msgstr "" + +#: src/Tax.php:278 src/Tax.php:336 +#, php-format +msgctxt "oss" +msgid "Tax class (%s)" +msgstr "" + +#: src/Tax.php:279 +msgctxt "oss" +msgid "Same as parent" +msgstr "" + +#: src/Tax.php:281 src/Tax.php:314 src/Tax.php:338 src/Tax.php:372 +msgctxt "oss" +msgid "remove" +msgstr "" + +#: src/Tax.php:291 src/Tax.php:349 +msgctxt "oss" +msgid "Add country specific tax class (OSS)" +msgstr "" + +#: src/Tax.php:298 src/Tax.php:356 +msgctxt "oss" +msgid "Select country" +msgstr "" + +#. translators: Do not translate +#: src/Tax.php:452 src/Tax.php:579 +msgid "Reduced rate" +msgstr "" + +#: src/Tax.php:455 src/Tax.php:571 +msgctxt "oss" +msgid "Greater reduced rate" +msgstr "" + +#: src/Tax.php:458 src/Tax.php:575 +msgctxt "oss" +msgid "Super reduced rate" +msgstr "" + +#: src/Tax.php:751 +msgctxt "oss" +msgid "Madeira" +msgstr "" + +#: src/Tax.php:758 +msgctxt "oss" +msgid "Acores" +msgstr "" + +#: src/Tax.php:794 +msgctxt "oss" +msgid "Northern Ireland" +msgstr "" + +#: src/Tax.php:807 +msgctxt "oss-tax-rate-import" +msgid "Exempt" +msgstr "" + +#: src/Tax.php:873 +#, php-format +msgctxt "oss-tax-rate-import" +msgid "VAT %1$s %% %2$s" +msgstr "" + +#. translators: %s: Customer billing full name +#: templates/emails/admin-delivery-threshold.php:19 +#, php-format +msgctxt "oss" +msgid "" +"Your OSS delivery threshold of %1$s has been reached. Please take action " +"immediately. Visit the OSS Settings Panel for details." +msgstr "" + +#: templates/emails/admin-delivery-threshold.php:21 +msgctxt "oss" +msgid "Report Details" +msgstr "" + +#: templates/emails/admin-delivery-threshold.php:24 +msgctxt "oss" +msgid "Period" +msgstr "" + +#: templates/emails/admin-delivery-threshold.php:29 +msgctxt "oss" +msgid "See report details" +msgstr "" + +#: templates/emails/plain/admin-delivery-threshold.php:17 +#, php-format +msgctxt "oss" +msgid "" +"Your OSS delivery threshold of %1$s has been reached. Please take action " +"immediately. Visit the OSS Settings Panel (%2$s) for details." +msgstr "" + +#. Plugin Name of the plugin/theme +msgid "One Stop Shop for WooCommerce" +msgstr "" + +#. Plugin URI of the plugin/theme +msgid "https://github.com/vendidero/one-stop-shop-woocommerce" +msgstr "" + +#. Description of the plugin/theme +msgid "Comply with the One Stop Shop procedure while using WooCommerce." +msgstr "" + +#. Author of the plugin/theme +msgid "vendidero" +msgstr "" + +#. Author URI of the plugin/theme +msgid "https://vendidero.de" +msgstr "" diff --git a/packages/one-stop-shop-woocommerce/libs/woocommerce-eu-tax-helper/src/Helper.php b/packages/one-stop-shop-woocommerce/libs/woocommerce-eu-tax-helper/src/Helper.php new file mode 100644 index 000000000..455d10ef8 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/libs/woocommerce-eu-tax-helper/src/Helper.php @@ -0,0 +1,911 @@ +countries->get_european_union_countries(); + + return $countries; + } + + public static function get_eu_vat_countries() { + return apply_filters( 'woocommerce_eu_tax_helper_eu_vat_countries', WC()->countries->get_european_union_countries( 'eu_vat' ) ); + } + + public static function is_northern_ireland( $country, $postcode = '' ) { + if ( 'GB' === $country && 'BT' === strtoupper( substr( trim( $postcode ), 0, 2 ) ) ) { + return true; + } elseif ( 'IX' === $country ) { + return true; + } + + return false; + } + + public static function is_eu_vat_country( $country, $postcode = '' ) { + $country = wc_strtoupper( $country ); + $postcode = wc_normalize_postcode( $postcode ); + $is_eu_vat_country = in_array( $country, self::get_eu_vat_countries(), true ); + + if ( self::is_northern_ireland( $country, $postcode ) ) { + $is_eu_vat_country = true; + } elseif ( self::is_eu_vat_postcode_exemption( $country, $postcode ) ) { + $is_eu_vat_country = false; + } + + return apply_filters( 'woocommerce_eu_tax_helper_is_eu_vat_country', $is_eu_vat_country, $country, $postcode ); + } + + public static function is_third_country( $country, $postcode = '' ) { + $is_third_country = true; + + /** + * In case the base country is within EU consider all non-EU VAT countries as third countries. + * In any other case consider every non-base-country as third country. + */ + if ( in_array( self::get_base_country(), self::get_eu_vat_countries(), true ) ) { + $is_third_country = ! self::is_eu_vat_country( $country, $postcode ); + } else { + $is_third_country = self::get_base_country() !== $country; + } + + return apply_filters( 'woocommerce_eu_tax_helper_is_third_country', $is_third_country, $country, $postcode ); + } + + public static function is_eu_country( $country ) { + return in_array( $country, self::get_eu_countries(), true ); + } + + public static function is_eu_vat_postcode_exemption( $country, $postcode = '' ) { + $country = wc_strtoupper( $country ); + $postcode = wc_normalize_postcode( $postcode ); + $exemptions = self::get_vat_postcode_exemptions_by_country(); + $is_exempt = false; + + if ( ! empty( $postcode ) && in_array( $country, self::get_eu_vat_countries(), true ) ) { + if ( array_key_exists( $country, $exemptions ) ) { + $wildcards = wc_get_wildcard_postcodes( $postcode, $country ); + + foreach ( $exemptions[ $country ] as $exempt_postcode ) { + if ( in_array( $exempt_postcode, $wildcards, true ) ) { + $is_exempt = true; + break; + } + } + } + } + + return $is_exempt; + } + + /** + * Get VAT exemptions (of EU countries) for certain postcodes (e.g. canary islands) + * + * @see https://www.hk24.de/produktmarken/beratung-service/recht-und-steuern/steuerrecht/umsatzsteuer-mehrwertsteuer/umsatzsteuer-mehrwertsteuer-international/verfahrensrecht/territoriale-besonderheiten-umsatzsteuer-zollrecht-1167674 + * @see https://github.com/woocommerce/woocommerce/issues/5143 + * @see https://ec.europa.eu/taxation_customs/business/vat/eu-vat-rules-topic/territorial-status-eu-countries-certain-territories_en + * + * @return \string[][] + */ + public static function get_vat_postcode_exemptions_by_country( $country = '' ) { + $country = wc_strtoupper( $country ); + + $exemptions = array( + 'DE' => array( + '27498', // Helgoland + '78266', // Büsingen am Hochrhein + ), + 'ES' => array( + '35*', // Canary Islands + '38*', // Canary Islands + '51*', // Ceuta + '52*', // Melilla + ), + 'GR' => array( + '63086', // Mount Athos + '63087', // Mount Athos + ), + 'FR' => array( + '971*', // Guadeloupe + '972*', // Martinique + '973*', // French Guiana + '974*', // Réunion + '976*', // Mayotte + ), + 'IT' => array( + '22060', // Livigno, Campione d’Italia + '23030', // Lake Lugano + ), + 'FI' => array( + '22*', // Aland islands + ), + ); + + if ( empty( $country ) ) { + return $exemptions; + } elseif ( array_key_exists( $country, $exemptions ) ) { + return $exemptions[ $country ]; + } else { + return array(); + } + } + + /** + * @param integer|\WC_Order $order + * + * @return array + */ + public static function get_order_taxable_location( $order ) { + $order = is_a( $order, 'WC_Order' ) ? $order : wc_get_order( $order ); + + $taxable_address = array( + WC()->countries->get_base_country(), + WC()->countries->get_base_state(), + WC()->countries->get_base_postcode(), + WC()->countries->get_base_city(), + ); + + if ( ! $order ) { + return $taxable_address; + } + + $tax_based_on = get_option( 'woocommerce_tax_based_on' ); + + if ( is_a( $order, 'WC_Order_Refund' ) ) { + $order = wc_get_order( $order->get_parent_id() ); + + if ( ! $order ) { + return $taxable_address; + } + } + + /** + * Shipping address data does not exist + */ + if ( 'shipping' === $tax_based_on && ! $order->get_shipping_country() ) { + $tax_based_on = 'billing'; + } + + $is_vat_exempt = apply_filters( 'woocommerce_order_is_vat_exempt', 'yes' === $order->get_meta( 'is_vat_exempt' ), $order ); + + /** + * In case the order is a VAT exempt, calculate net prices based on taxes from base country. + */ + if ( $is_vat_exempt ) { + $tax_based_on = 'base'; + } + + $country = 'shipping' === $tax_based_on ? $order->get_shipping_country() : $order->get_billing_country(); + + if ( 'base' !== $tax_based_on && ! empty( $country ) ) { + $taxable_address = array( + $country, + 'billing' === $tax_based_on ? $order->get_billing_state() : $order->get_shipping_state(), + 'billing' === $tax_based_on ? $order->get_billing_postcode() : $order->get_shipping_postcode(), + 'billing' === $tax_based_on ? $order->get_billing_city() : $order->get_shipping_city(), + ); + } + + return $taxable_address; + } + + public static function get_taxable_location() { + $is_admin_order_request = self::is_admin_order_request(); + + if ( $is_admin_order_request ) { + $taxable_address = array( + WC()->countries->get_base_country(), + WC()->countries->get_base_state(), + WC()->countries->get_base_postcode(), + WC()->countries->get_base_city(), + ); + + if ( isset( $_POST['order_id'] ) && ( $order = wc_get_order( absint( $_POST['order_id'] ) ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $taxable_address = self::get_order_taxable_location( $order ); + } + + return $taxable_address; + } else { + return \WC_Tax::get_tax_location(); + } + } + + public static function is_admin_order_ajax_request() { + $order_actions = array( 'woocommerce_calc_line_taxes', 'woocommerce_save_order_items', 'add_coupon_discount', 'refund_line_items', 'delete_refund' ); + + return isset( $_POST['action'], $_POST['order_id'] ) && ( strstr( wc_clean( wp_unslash( $_POST['action'] ) ), '_order_' ) || in_array( wc_clean( wp_unslash( $_POST['action'] ) ), $order_actions, true ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + public static function is_admin_order_request() { + return is_admin() && current_user_can( 'edit_shop_orders' ) && self::is_admin_order_ajax_request(); + } + + public static function current_request_has_vat_exempt() { + $is_admin_order_request = self::is_admin_order_request(); + $is_vat_exempt = false; + + if ( $is_admin_order_request ) { + if ( $order = wc_get_order( absint( $_POST['order_id'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing,WordPress.Security.ValidatedSanitizedInput.InputNotValidated + $is_vat_exempt = apply_filters( 'woocommerce_order_is_vat_exempt', 'yes' === $order->get_meta( 'is_vat_exempt' ), $order ); + } + } else { + if ( WC()->customer && WC()->customer->is_vat_exempt() ) { + $is_vat_exempt = true; + } + } + + return $is_vat_exempt; + } + + public static function get_base_country() { + if ( WC()->countries ) { + return WC()->countries->get_base_country(); + } else { + return wc_get_base_location()['country']; + } + } + + /** + * Returns a list of EU countries except base country. + * + * @return string[] + */ + public static function get_non_base_eu_countries( $include_gb = false ) { + $countries = WC()->countries->get_european_union_countries( 'eu_vat' ); + + /** + * Include GB to allow Northern Ireland + */ + if ( $include_gb && ! in_array( 'GB', $countries, true ) ) { + $countries = array_merge( $countries, array( 'GB' ) ); + } + + $base_country = self::get_base_country(); + $countries = array_diff( $countries, array( $base_country ) ); + + return $countries; + } + + public static function country_supports_eu_vat( $country, $postcode = '' ) { + return self::is_eu_vat_country( $country, $postcode ); + } + + public static function import_oss_tax_rates( $tax_class_slug_names = array() ) { + self::import_tax_rates_internal( true, $tax_class_slug_names ); + } + + public static function import_default_tax_rates( $tax_class_slug_names = array() ) { + self::import_tax_rates_internal( false, $tax_class_slug_names ); + } + + public static function import_tax_rates( $tax_class_slug_names = array() ) { + self::import_tax_rates_internal( self::oss_procedure_is_enabled(), $tax_class_slug_names ); + } + + protected static function parse_tax_class_slug_names( $tax_class_slug_names = array() ) { + return wp_parse_args( + $tax_class_slug_names, + array( + 'reduced' => apply_filters( 'woocommerce_eu_tax_helper_tax_class_reduced_name', __( 'Reduced rate', 'woocommerce' ) ), // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + 'greater-reduced' => apply_filters( 'woocommerce_eu_tax_helper_tax_class_greater_reduced_name', _x( 'Greater reduced rate', 'tax-helper-tax-class-name', 'oss-woocommerce' ) ), + 'super-reduced' => apply_filters( 'woocommerce_eu_tax_helper_tax_class_super_reduced_name', _x( 'Super reduced rate', 'tax-helper-tax-class-name', 'oss-woocommerce' ) ), + ) + ); + } + + protected static function import_tax_rates_internal( $is_oss = true, $tax_class_slug_names = array() ) { + self::clear_cache(); + + $tax_class_slugs = self::get_tax_class_slugs( $tax_class_slug_names ); + $tax_class_slug_names = self::parse_tax_class_slug_names( $tax_class_slug_names ); + $eu_rates = self::get_eu_tax_rates(); + + foreach ( $tax_class_slugs as $tax_class_type => $class ) { + /** + * Maybe create missing tax classes + */ + if ( false === $class ) { + switch ( $tax_class_type ) { + case 'reduced': + /* translators: Do not translate */ + \WC_Tax::create_tax_class( $tax_class_slug_names['reduced'] ); // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + break; + case 'greater-reduced': + \WC_Tax::create_tax_class( $tax_class_slug_names['greater-reduced'] ); + break; + case 'super-reduced': + \WC_Tax::create_tax_class( $tax_class_slug_names['super-reduced'] ); + break; + } + } + + $new_rates = array(); + + foreach ( $eu_rates as $country => $rates_data ) { + + /** + * Use base country rates in case OSS is disabled + */ + if ( ! $is_oss ) { + $base_country = self::get_base_country(); + + if ( isset( $eu_rates[ $base_country ] ) ) { + /** + * In case the country includes multiple rules (e.g. postcode exempts) by default + * do only use the last rule (which does not include exempts) to construct non-base country tax rules. + */ + if ( $base_country !== $country ) { + $base_country_base_rate = array_values( array_slice( $eu_rates[ $base_country ], -1 ) )[0]; + + foreach ( $rates_data as $key => $rate_data ) { + $rates_data[ $key ] = array_replace_recursive( $rate_data, $base_country_base_rate ); + + foreach ( $tax_class_slugs as $tmp_class_type => $class_data ) { + /** + * Do not include tax classes which are not supported by the base country. + */ + if ( isset( $rates_data[ $key ][ $tmp_class_type ] ) && ! isset( $base_country_base_rate[ $tmp_class_type ] ) ) { + unset( $rates_data[ $key ][ $tmp_class_type ] ); + } elseif ( isset( $rates_data[ $key ][ $tmp_class_type ] ) ) { + /** + * Replace tax class data with base data to make sure that reduced + * classes have the same dimensions + */ + $rates_data[ $key ][ $tmp_class_type ] = $base_country_base_rate[ $tmp_class_type ]; + + /** + * In case this is an exempt make sure to replace with zero tax rates + */ + if ( isset( $rate_data['is_exempt'] ) && $rate_data['is_exempt'] ) { + if ( is_array( $rates_data[ $key ][ $tmp_class_type ] ) ) { + foreach ( $rates_data[ $key ][ $tmp_class_type ] as $k => $rate ) { + $rates_data[ $key ][ $tmp_class_type ][ $k ] = 0; + } + } else { + $rates_data[ $key ][ $tmp_class_type ] = 0; + } + } + } + } + } + } + } else { + continue; + } + } + + /** + * Each country may contain multiple tax rates + */ + foreach ( $rates_data as $rates ) { + + $rates = wp_parse_args( + $rates, + array( + 'name' => '', + 'postcodes' => array(), + 'reduced' => array(), + ) + ); + + if ( ! empty( $rates['postcode'] ) ) { + foreach ( $rates['postcode'] as $postcode ) { + $tax_rate = self::get_single_tax_rate_data( $tax_class_type, $rates, $country, $postcode ); + + if ( false !== $tax_rate ) { + $new_rates[] = $tax_rate; + } + } + } else { + $tax_rate = self::get_single_tax_rate_data( $tax_class_type, $rates, $country ); + + if ( false !== $tax_rate ) { + $new_rates[] = $tax_rate; + } + } + } + } + + self::import_rates( $new_rates, $class ); + } + } + + private static function get_single_tax_rate_data( $tax_class_type, $rates, $country, $postcode = '' ) { + $rates = wp_parse_args( + $rates, + array( + 'name' => '', + 'reduced' => array(), + ) + ); + + $single_rate = array( + 'name' => $rates['name'], + 'rate' => false, + 'country' => $country, + 'postcode' => $postcode, + ); + + switch ( $tax_class_type ) { + case 'greater-reduced': + if ( count( $rates['reduced'] ) > 1 ) { + $single_rate['rate'] = $rates['reduced'][1]; + } + break; + case 'reduced': + if ( ! empty( $rates['reduced'] ) ) { + $single_rate['rate'] = $rates['reduced'][0]; + } + break; + default: + if ( isset( $rates[ $tax_class_type ] ) ) { + $single_rate['rate'] = $rates[ $tax_class_type ]; + } + break; + } + + if ( false === $single_rate['rate'] ) { + return false; + } + + return $single_rate; + } + + protected static function clear_cache() { + $cache_key = \WC_Cache_Helper::get_cache_prefix( 'taxes' ) . 'eu_tax_helper_tax_class_slugs'; + + wp_cache_delete( $cache_key, 'taxes' ); + } + + public static function get_tax_class_slugs( $tax_class_slug_names = array() ) { + $tax_class_slug_names = self::parse_tax_class_slug_names( $tax_class_slug_names ); + $cache_key = \WC_Cache_Helper::get_cache_prefix( 'taxes' ) . 'eu_tax_helper_tax_class_slugs'; + $slugs = wp_cache_get( $cache_key, 'taxes' ); + + if ( false === $slugs ) { + $reduced_tax_class = false; + $greater_reduced_tax_class = false; + $super_reduced_tax_class = false; + $tax_classes = \WC_Tax::get_tax_class_slugs(); + + /** + * Try to determine the reduced tax rate class + */ + foreach ( $tax_classes as $slug ) { + if ( strstr( $slug, 'virtual' ) ) { + continue; + } + + if ( ! $greater_reduced_tax_class && strstr( $slug, sanitize_title( 'Greater reduced rate' ) ) ) { + $greater_reduced_tax_class = $slug; + } elseif ( ! $greater_reduced_tax_class && strstr( $slug, sanitize_title( $tax_class_slug_names['greater-reduced'] ) ) ) { + $greater_reduced_tax_class = $slug; + } elseif ( ! $super_reduced_tax_class && strstr( $slug, sanitize_title( 'Super reduced rate' ) ) ) { + $super_reduced_tax_class = $slug; + } elseif ( ! $super_reduced_tax_class && strstr( $slug, sanitize_title( $tax_class_slug_names['super-reduced'] ) ) ) { + $super_reduced_tax_class = $slug; + } elseif ( ! $reduced_tax_class && strstr( $slug, sanitize_title( 'Reduced rate' ) ) ) { + $reduced_tax_class = $slug; + } elseif ( ! $reduced_tax_class && strstr( $slug, sanitize_title( $tax_class_slug_names['reduced'] ) ) ) { // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + $reduced_tax_class = $slug; + } elseif ( ! $reduced_tax_class && strstr( $slug, 'reduced' ) && ! $reduced_tax_class ) { + $reduced_tax_class = $slug; + } + } + + $slugs = array( + 'reduced' => $reduced_tax_class, + 'greater-reduced' => $greater_reduced_tax_class, + 'super-reduced' => $super_reduced_tax_class, + 'standard' => '', + ); + + wp_cache_set( $cache_key, $slugs, 'taxes' ); + } + + return apply_filters( 'woocommerce_eu_tax_helper_tax_rate_class_slugs', $slugs ); + } + + public static function get_tax_type_by_country_rate( $rate_percentage, $country ) { + $country = strtoupper( $country ); + + /** + * Map northern ireland to GB + */ + if ( 'XI' === $country ) { + $country = 'GB'; + } + + $eu_rates = self::get_eu_tax_rates(); + $tax_type = 'standard'; + + if ( array_key_exists( $country, $eu_rates ) ) { + $rates = $eu_rates[ $country ]; + + foreach ( $rates as $rate ) { + foreach ( $rate as $tax_rate_type => $tax_rate_percent ) { + if ( ( is_array( $tax_rate_percent ) && in_array( $rate_percentage, $tax_rate_percent, true ) ) || (float) $tax_rate_percent === (float) $rate_percentage ) { + $tax_type = $tax_rate_type; + break; + } + } + } + } + + return apply_filters( 'woocommerce_eu_tax_helper_country_rate_tax_type', $tax_type, $country, $rate_percentage ); + } + + public static function get_eu_tax_rates() { + /** + * @see https://europa.eu/youreurope/business/taxation/vat/vat-rules-rates/index_en.htm + * + * Include Great Britain to allow including Norther Ireland + */ + $rates = array( + 'AT' => array( + array( + 'standard' => 20, + 'reduced' => array( 10, 13 ), + ), + ), + 'BE' => array( + array( + 'standard' => 21, + 'reduced' => array( 6, 12 ), + ), + ), + 'BG' => array( + array( + 'standard' => 20, + 'reduced' => array( 9 ), + ), + ), + 'CY' => array( + array( + 'standard' => 19, + 'reduced' => array( 5, 9 ), + ), + ), + 'CZ' => array( + array( + 'standard' => 21, + 'reduced' => array( 10, 15 ), + ), + ), + 'DE' => array( + array( + 'standard' => 19, + 'reduced' => array( 7 ), + ), + ), + 'DK' => array( + array( + 'standard' => 25, + 'reduced' => array(), + ), + ), + 'EE' => array( + array( + 'standard' => 20, + 'reduced' => array( 9 ), + ), + ), + 'GR' => array( + array( + 'standard' => 24, + 'reduced' => array( 6, 13 ), + ), + ), + 'ES' => array( + array( + 'standard' => 21, + 'reduced' => array( 10 ), + 'super-reduced' => 4, + ), + ), + 'FI' => array( + array( + 'standard' => 24, + 'reduced' => array( 10, 14 ), + ), + ), + 'FR' => array( + array( + 'standard' => 20, + 'reduced' => array( 5.5, 10 ), + 'super-reduced' => 2.1, + ), + ), + 'HR' => array( + array( + 'standard' => 25, + 'reduced' => array( 5, 13 ), + ), + ), + 'HU' => array( + array( + 'standard' => 27, + 'reduced' => array( 5, 18 ), + ), + ), + 'IE' => array( + array( + 'standard' => 23, + 'reduced' => array( 9, 13.5 ), + 'super-reduced' => 4.8, + ), + ), + 'IT' => array( + array( + 'standard' => 22, + 'reduced' => array( 5, 10 ), + 'super-reduced' => 4, + ), + ), + 'LT' => array( + array( + 'standard' => 21, + 'reduced' => array( 5, 9 ), + ), + ), + 'LU' => array( + array( + 'standard' => 16, + 'reduced' => array( 7 ), + 'super-reduced' => 3, + ), + ), + 'LV' => array( + array( + 'standard' => 21, + 'reduced' => array( 12, 5 ), + ), + ), + 'MC' => array( + array( + 'standard' => 20, + 'reduced' => array( 5.5, 10 ), + 'super-reduced' => 2.1, + ), + ), + 'MT' => array( + array( + 'standard' => 18, + 'reduced' => array( 5, 7 ), + ), + ), + 'NL' => array( + array( + 'standard' => 21, + 'reduced' => array( 9 ), + ), + ), + 'PL' => array( + array( + 'standard' => 23, + 'reduced' => array( 5, 8 ), + ), + ), + 'PT' => array( + array( + // Madeira + 'postcode' => array( '90*', '91*', '92*', '93*', '94*' ), + 'standard' => 22, + 'reduced' => array( 5, 12 ), + 'name' => _x( 'Madeira', 'tax-helper', 'oss-woocommerce' ), + ), + array( + // Acores + 'postcode' => array( '95*', '96*', '97*', '98*', '99*' ), + 'standard' => 18, + 'reduced' => array( 4, 9 ), + 'name' => _x( 'Acores', 'tax-helper', 'oss-woocommerce' ), + ), + array( + 'standard' => 23, + 'reduced' => array( 6, 13 ), + ), + ), + 'RO' => array( + array( + 'standard' => 19, + 'reduced' => array( 5, 9 ), + ), + ), + 'SE' => array( + array( + 'standard' => 25, + 'reduced' => array( 6, 12 ), + ), + ), + 'SI' => array( + array( + 'standard' => 22, + 'reduced' => array( 9.5 ), + ), + ), + 'SK' => array( + array( + 'standard' => 20, + 'reduced' => array( 10 ), + ), + ), + 'GB' => array( + array( + 'standard' => 20, + 'reduced' => array( 5 ), + 'postcode' => array( 'BT*' ), + 'name' => _x( 'Northern Ireland', 'tax-helper', 'oss-woocommerce' ), + ), + ), + ); + + foreach ( self::get_vat_postcode_exemptions_by_country() as $country => $exempt_postcodes ) { + if ( array_key_exists( $country, $rates ) ) { + $default_rate = array_values( $rates[ $country ] )[0]; + + $postcode_exempt = array( + 'postcode' => $exempt_postcodes, + 'standard' => 0, + 'reduced' => count( $default_rate['reduced'] ) > 1 ? array( 0, 0 ) : array( 0 ), + 'name' => _x( 'Exempt', 'tax-helper-rate-import', 'oss-woocommerce' ), + 'is_exempt' => true, + ); + + if ( array_key_exists( 'super-reduced', $default_rate ) ) { + $postcode_exempt['super-reduced'] = 0; + } + + // Prepend before other tax rates + $rates[ $country ] = array_merge( array( $postcode_exempt ), $rates[ $country ] ); + } + } + + return $rates; + } + + /** + * @param \stdClass $rate + * + * @return bool + */ + public static function tax_rate_is_northern_ireland( $rate ) { + if ( 'GB' === $rate->tax_rate_country && isset( $rate->postcode ) && ! empty( $rate->postcode ) ) { + foreach ( $rate->postcode as $postcode ) { + if ( self::is_northern_ireland( $rate->tax_rate_country, $postcode ) ) { + return true; + } + } + } + + return false; + } + + public static function import_rates( $rates, $tax_class = '' ) { + global $wpdb; + + $eu_countries = self::get_eu_vat_countries(); + + /** + * Delete EU tax rates and make sure tax rate locations are deleted too + */ + foreach ( \WC_Tax::get_rates_for_tax_class( $tax_class ) as $rate_id => $rate ) { + if ( in_array( $rate->tax_rate_country, $eu_countries, true ) || self::tax_rate_is_northern_ireland( $rate ) || ( 'GB' === $rate->tax_rate_country && 'GB' !== self::get_base_country() ) ) { + \WC_Tax::_delete_tax_rate( $rate_id ); + } + } + + $count = 0; + + foreach ( $rates as $rate ) { + $rate = wp_parse_args( + $rate, + array( + 'rate' => 0, + 'country' => '', + 'postcode' => '', + 'name' => '', + ) + ); + + $iso = wc_strtoupper( $rate['country'] ); + $vat_desc = $iso; + + if ( ! empty( $rate['name'] ) ) { + $vat_desc = $vat_desc . ' ' . $rate['name']; + } + + $vat_rate = wc_format_decimal( $rate['rate'], false, true ); + + $tax_rate_name = apply_filters( 'woocommerce_eu_tax_helper_import_tax_rate_name', sprintf( _x( 'VAT %1$s %% %2$s', 'tax-helper-rate-import', 'oss-woocommerce' ), $vat_rate, $vat_desc ), $rate['rate'], $iso, $tax_class, $rate ); + + $_tax_rate = array( + 'tax_rate_country' => $iso, + 'tax_rate_state' => '', + 'tax_rate' => (string) number_format( (float) wc_clean( $rate['rate'] ), 4, '.', '' ), + 'tax_rate_name' => $tax_rate_name, + 'tax_rate_compound' => 0, + 'tax_rate_priority' => 1, + 'tax_rate_order' => $count++, + 'tax_rate_shipping' => ( strstr( $tax_class, 'virtual' ) ? 0 : 1 ), + 'tax_rate_class' => $tax_class, + ); + + $new_tax_rate_id = \WC_Tax::_insert_tax_rate( $_tax_rate ); + + if ( ! empty( $rate['postcode'] ) ) { + \WC_Tax::_update_tax_rate_postcodes( $new_tax_rate_id, $rate['postcode'] ); + } + } + } + + /** + * @param $rate_id + * @param \WC_Order $order + */ + public static function get_tax_rate_percent( $rate_id, $order ) { + $taxes = $order->get_taxes(); + $percentage = null; + + foreach ( $taxes as $tax ) { + if ( (int) $tax->get_rate_id() === (int) $rate_id ) { + if ( is_callable( array( $tax, 'get_rate_percent' ) ) ) { + $percentage = $tax->get_rate_percent(); + } + } + } + + /** + * WC_Order_Item_Tax::get_rate_percent returns null by default. + * Fallback to global tax rates (DB) in case the percentage is not available within order data. + */ + if ( is_null( $percentage ) || '' === $percentage ) { + $rate_percentage = self::get_tax_rate_percentage( $rate_id ); + + if ( false !== $rate_percentage ) { + $percentage = $rate_percentage; + } + } + + if ( ! is_numeric( $percentage ) ) { + $percentage = 0; + } + + return $percentage; + } + + public static function get_tax_rate_percentage( $rate_id ) { + $percentage = false; + + if ( is_callable( array( 'WC_Tax', 'get_rate_percent_value' ) ) ) { + $percentage = \WC_Tax::get_rate_percent_value( $rate_id ); + } elseif ( is_callable( array( 'WC_Tax', 'get_rate_percent' ) ) ) { + $percentage = filter_var( \WC_Tax::get_rate_percent( $rate_id ), FILTER_SANITIZE_NUMBER_FLOAT, FILTER_FLAG_ALLOW_FRACTION ); + } + + return $percentage; + } +} diff --git a/packages/one-stop-shop-woocommerce/license.txt b/packages/one-stop-shop-woocommerce/license.txt new file mode 100644 index 000000000..76adc409b --- /dev/null +++ b/packages/one-stop-shop-woocommerce/license.txt @@ -0,0 +1,699 @@ +One Stop Shop WooCommerce + +Copyright 2011 by the contributors + +This program is free software; you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation; either version 3 of the License, or +(at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with this program; if not, write to the Free Software +Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + +This program incorporates work covered by the following copyright and +permission notices: + + WooCommerce + +=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= + + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright © 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright © + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright © + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. \ No newline at end of file diff --git a/packages/one-stop-shop-woocommerce/one-stop-shop-woocommerce.php b/packages/one-stop-shop-woocommerce/one-stop-shop-woocommerce.php new file mode 100644 index 000000000..9834787fa --- /dev/null +++ b/packages/one-stop-shop-woocommerce/one-stop-shop-woocommerce.php @@ -0,0 +1,77 @@ + +
+

+ composer install', + '' . esc_html( str_replace( ABSPATH, '', __DIR__ ) ) . '' + ); + ?> +

+
+ get_name() === $wc_admin_note_name ) { + /** + * Update notice hide in case note has been actioned (e.g. button click by user) + */ + if ( Note::E_WC_ADMIN_NOTE_ACTIONED === $note->get_status() ) { + update_option( 'oss_hide_notice_' . sanitize_key( $oss_note::get_id() ), 'yes' ); + } + + break; + } + } + } + } catch ( \Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } + } + + public static function register_tax_rate_refresh_tool( $tools ) { + $tools['refresh_oss_tax_rates'] = array( + 'name' => _x( 'Refresh VAT rates (OSS)', 'oss', 'woocommerce-germanized' ), + 'button' => _x( 'Refresh VAT rates (OSS)', 'oss', 'woocommerce-germanized' ), + 'callback' => array( __CLASS__, 'refresh_vat_rates' ), + 'desc' => sprintf( + '%1$s %2$s', + _x( 'Note:', 'oss', 'woocommerce-germanized' ), + sprintf( _x( 'This option will delete all of your current EU VAT rates and re-import them based on your current OSS status.', 'oss', 'woocommerce-germanized' ), esc_url( Settings::get_settings_url() ) ) + ), + ); + + return $tools; + } + + public static function refresh_vat_rates() { + if ( Helper::oss_procedure_is_enabled() ) { + Helper::import_oss_tax_rates(); + } else { + Helper::import_default_tax_rates(); + } + } + + public static function html_field( $value ) { + ?> + + + + + + + id : ''; + $supports_notes = self::supports_wc_admin(); + + if ( ! $supports_notes || in_array( $screen_id, array( 'dashboard', 'plugins' ), true ) ) { + foreach ( self::get_notes() as $note ) { + if ( $note::is_enabled() ) { + $note::render(); + } + } + } + } + + /** + * @return AdminNote[] + */ + public static function get_notes() { + $notes = array( 'Vendidero\OneStopShop\DeliveryThresholdWarning' ); + + if ( ! Package::enable_auto_observer() ) { + $notes = array(); + } + + return $notes; + } + + public static function supports_wc_admin() { + $supports_notes = class_exists( 'Automattic\WooCommerce\Admin\Notes\Note' ); + + try { + $data_store = \WC_Data_Store::load( 'admin-note' ); + } catch ( \Exception $e ) { + $supports_notes = false; + } + + return $supports_notes; + } + + protected static function get_wc_admin_note_name( $oss_note_id ) { + return 'oss_' . $oss_note_id; + } + + protected static function get_wc_admin_note( $oss_note_id ) { + $note_name = self::get_wc_admin_note_name( $oss_note_id ); + $data_store = \WC_Data_Store::load( 'admin-note' ); + $note_ids = $data_store->get_notes_with_name( $note_name ); + + if ( ! empty( $note_ids ) && ( $note = Notes::get_note( $note_ids[0] ) ) ) { + return $note; + } + + return false; + } + + public static function queue_wc_admin_notes() { + if ( self::supports_wc_admin() ) { + foreach ( self::get_notes() as $oss_note ) { + $note = self::get_wc_admin_note( $oss_note::get_id() ); + + if ( ! $note && $oss_note::is_enabled() ) { + $note = new Note(); + $note->set_title( $oss_note::get_title() ); + $note->set_content( $oss_note::get_content() ); + $note->set_content_data( (object) array() ); + $note->set_type( 'update' ); + $note->set_name( self::get_wc_admin_note_name( $oss_note::get_id() ) ); + $note->set_source( 'oss-woocommerce' ); + $note->set_status( Note::E_WC_ADMIN_NOTE_UNACTIONED ); + + foreach ( $oss_note::get_actions() as $action ) { + $note->add_action( + 'oss_' . sanitize_key( $action['title'] ), + $action['title'], + $action['url'], + Note::E_WC_ADMIN_NOTE_ACTIONED, + $action['is_primary'] ? true : false + ); + } + + $note->save(); + } elseif ( $oss_note::is_enabled() && $note ) { + $note->set_status( Note::E_WC_ADMIN_NOTE_UNACTIONED ); + $note->save(); + } + } + } + } + + public static function get_threshold_notice_content() { + return sprintf( _x( 'Seems like you have reached (or are close to reaching) the delivery threshold for the current year. Please make sure to check the report details and take action in case necessary.', 'oss', 'woocommerce-germanized' ), esc_url( Package::get_observer_report()->get_url() ) ); + } + + public static function get_threshold_notice_title() { + return _x( 'Delivery threshold reached (OSS)', 'oss', 'woocommerce-germanized' ); + } + + public static function init_observer() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'oss_init_observer' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + if ( ! Queue::get_running_observer() ) { + Package::update_observer_report(); + } + + wp_safe_redirect( wp_get_referer() ); + exit(); + } + + public static function switch_procedure() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'oss_switch_procedure' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + if ( Helper::oss_procedure_is_enabled() ) { + update_option( 'oss_use_oss_procedure', 'no' ); + + Helper::import_default_tax_rates(); + + do_action( 'woocommerce_oss_disabled_oss_procedure' ); + } else { + update_option( 'woocommerce_tax_based_on', 'shipping' ); + update_option( 'oss_use_oss_procedure', 'yes' ); + + Helper::import_oss_tax_rates(); + + do_action( 'woocommerce_oss_enabled_oss_procedure' ); + } + + do_action( 'woocommerce_oss_switched_oss_procedure_status' ); + + wp_safe_redirect( wp_get_referer() ); + exit(); + } + + public static function hide_notice() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'oss_hide_notice' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $notice_id = isset( $_GET['notice'] ) ? wc_clean( wp_unslash( $_GET['notice'] ) ) : ''; + + foreach ( self::get_notes() as $oss_note ) { + if ( $oss_note::get_id() === $notice_id ) { + update_option( 'oss_hide_notice_' . sanitize_key( $oss_note::get_id() ), 'yes' ); + + if ( self::supports_wc_admin() ) { + self::delete_wc_admin_note( $oss_note ); + } + + break; + } + } + + wp_safe_redirect( wp_get_referer() ); + exit(); + } + + /** + * @param AdminNote $oss_note + */ + public static function delete_wc_admin_note( $oss_note ) { + if ( ! self::supports_wc_admin() ) { + return false; + } + + try { + if ( $note = self::get_wc_admin_note( $oss_note::get_id() ) ) { + $note->delete( true ); + return true; + } + + return false; + } catch ( \Exception $e ) { + return false; + } + } + + public static function delete_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'oss_delete_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_id = isset( $_GET['report_id'] ) ? wc_clean( wp_unslash( $_GET['report_id'] ) ) : ''; + + if ( ! empty( $report_id ) && ( $report = Package::get_report( $report_id ) ) ) { + $report->delete(); + + $referer = self::get_clean_referer(); + + /** + * Do not redirect deleted, refreshed reports back to report details page + */ + if ( strstr( $referer, '&report=' ) ) { + $referer = admin_url( 'admin.php?page=oss-reports' ); + } + + wp_safe_redirect( esc_url_raw( add_query_arg( array( 'report_deleted' => $report_id ), $referer ) ) ); + exit(); + } + + wp_safe_redirect( esc_url_raw( wp_get_referer() ) ); + exit(); + } + + protected static function get_clean_referer() { + $referer = wp_get_referer(); + + return remove_query_arg( array( 'report_created', 'report_deleted', 'report_restarted', 'report_cancelled' ), $referer ); + } + + public static function export_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'oss_export_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_id = isset( $_GET['report_id'] ) ? wc_clean( wp_unslash( $_GET['report_id'] ) ) : ''; + $decimals = isset( $_GET['decimals'] ) ? absint( wp_unslash( $_GET['decimals'] ) ) : wc_get_price_decimals(); + $export_type = isset( $_GET['export_type'] ) ? wc_clean( wp_unslash( $_GET['export_type'] ) ) : ''; + + if ( ! empty( $report_id ) && ( $report = Package::get_report( $report_id ) ) ) { + $exporter_class = '\Vendidero\OneStopShop\CSVExporter'; + $base_country = Helper::get_base_country(); + + if ( 'bop' === $export_type ) { + $exporter_class = '\Vendidero\OneStopShop\CSVExporterBOP'; + } + + $exporter_class = apply_filters( 'oss_csv_exporter_classname', $exporter_class, $base_country ); + + if ( ! class_exists( $exporter_class ) ) { + $exporter_class = '\Vendidero\OneStopShop\CSVExporter'; + } + + $csv = new $exporter_class( $report_id, $decimals ); + $csv->export(); + } else { + wp_safe_redirect( wp_get_referer() ); + exit(); + } + } + + public static function refresh_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'oss_refresh_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_id = isset( $_GET['report_id'] ) ? wc_clean( wp_unslash( $_GET['report_id'] ) ) : ''; + + if ( ! empty( $report_id ) && ( $report = Package::get_report( $report_id ) ) ) { + Queue::start( $report->get_type(), $report->get_date_start(), $report->get_date_end() ); + + wp_safe_redirect( esc_url_raw( add_query_arg( array( 'report_restarted' => $report_id ), self::get_clean_referer() ) ) ); + exit(); + } + + wp_safe_redirect( esc_url_raw( wp_get_referer() ) ); + exit(); + } + + public static function cancel_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'oss_cancel_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_id = isset( $_GET['report_id'] ) ? wc_clean( wp_unslash( $_GET['report_id'] ) ) : ''; + + if ( ! empty( $report_id ) && Queue::is_running( $report_id ) ) { + Queue::cancel( $report_id ); + + $referer = self::get_clean_referer(); + + /** + * Do not redirect deleted, refreshed reports back to report details page + */ + if ( strstr( $referer, '&report=' ) ) { + $referer = admin_url( 'admin.php?page=oss-reports' ); + } + + wp_safe_redirect( esc_url_raw( add_query_arg( array( 'report_cancelled' => $report_id ), $referer ) ) ); + exit(); + } + + wp_safe_redirect( esc_url_raw( wp_get_referer() ) ); + exit(); + } + + public static function create_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_POST['_wpnonce'] ) ? wp_unslash( $_POST['_wpnonce'] ) : '', 'oss_create_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_type = ! empty( $_POST['report_type'] ) ? wc_clean( wp_unslash( $_POST['report_type'] ) ) : 'yearly'; + $report_type = array_key_exists( $report_type, Package::get_available_report_types() ) ? $report_type : 'yearly'; + $start_date = null; + $end_date = null; + + if ( 'quarterly' === $report_type ) { + $start_date = ! empty( $_POST['report_quarter'] ) ? wc_clean( wp_unslash( $_POST['report_quarter'] ) ) : null; + } elseif ( 'yearly' === $report_type ) { + $start_date = ! empty( $_POST['report_year'] ) ? wc_clean( wp_unslash( $_POST['report_year'] ) ) : null; + } elseif ( 'monthly' === $report_type ) { + $start_date = ! empty( $_POST['report_month'] ) ? wc_clean( wp_unslash( $_POST['report_month'] ) ) : null; + } elseif ( 'custom' === $report_type ) { + $start_date = ! empty( $_POST['date_start'] ) ? wc_clean( wp_unslash( $_POST['date_start'] ) ) : null; + $end_date = ! empty( $_POST['date_end'] ) ? wc_clean( wp_unslash( $_POST['date_end'] ) ) : null; + } + + if ( ! is_null( $start_date ) ) { + $start_date = Package::string_to_datetime( $start_date ); + } + + if ( ! is_null( $end_date ) ) { + $end_date = Package::string_to_datetime( $end_date ); + } + + $generator_id = Queue::start( $report_type, $start_date, $end_date ); + + wp_safe_redirect( admin_url( 'admin.php?page=oss-reports&report_created=' . $generator_id ) ); + exit(); + } + + public static function add_menu() { + add_submenu_page( 'woocommerce', _x( 'OSS', 'oss', 'woocommerce-germanized' ), _x( 'One Stop Shop', 'oss', 'woocommerce-germanized' ), 'manage_woocommerce', 'oss-reports', array( __CLASS__, 'render_report_page' ) ); + } + + protected static function render_create_report() { + $years = array(); + $years[] = date( 'Y' ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + $years[] = date( 'Y', strtotime( '-1 year' ) ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + + $quarters_selectable = array(); + $years_selectable = array(); + $months_selectable = array(); + + foreach ( $years as $year ) { + $start_day = date( 'Y-m-d', strtotime( $year . '-01-01' ) ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + $years_selectable[ $start_day ] = $year; + + for ( $i = 4; $i >= 1; $i-- ) { + $start_month = ( $i - 1 ) * 3 + 1; + $start_day = date( 'Y-m-d', strtotime( $year . '-' . $start_month . '-01' ) ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + + if ( date( 'Y-m-d' ) >= $start_day ) { // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + $quarters_selectable[ $start_day ] = sprintf( _x( 'Q%1$s/%2$s', 'oss', 'woocommerce-germanized' ), $i, $year ); + } + } + + for ( $i = 12; $i >= 1; $i-- ) { + $start_day = date( 'Y-m-d', strtotime( $year . '-' . $i . '-01' ) ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + $month = date( 'm', strtotime( $year . '-' . $i . '-01' ) ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + + if ( date( 'Y-m-d' ) >= $start_day ) { // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + $months_selectable[ $start_day ] = sprintf( _x( '%1$s/%2$s', 'oss', 'woocommerce-germanized' ), $month, $year ); + } + } + } + ?> +
+
+
+

+
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+ + + +
+ + + +
+ + + +
+ + + + + +
+
+
+ +
+ + +
+
+ +
+

+ + +
+ + output_notices(); + $_SERVER['REQUEST_URI'] = remove_query_arg( array( 'updated', 'changed', 'deleted', 'trashed', 'untrashed' ), ( isset( $_SERVER['REQUEST_URI'] ) ? esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) : admin_url( 'admin.php?page=oss-reports' ) ) ); + ?> + + views(); ?> + +
+ + + display(); ?> +
+ +
+
+
+ $action ) { + if ( isset( $action['url'] ) ) { + $target = isset( $action['target'] ) ? $action['target'] : '_self'; + + printf( '%5$s', esc_attr( $action_name ), esc_url( $action['url'] ), ( ( isset( $action['title'] ) ) ? esc_attr( $action['title'] ) : esc_attr( $action_name ) ), esc_attr( $target ), ( isset( $action['title'] ) ? esc_html( $action['title'] ) : esc_html( $action_name ) ) ); + } + } + } + + /** + * @param Report $report + * + * @return array[] + */ + public static function get_report_actions( $report ) { + $actions = array( + 'view' => array( + 'url' => $report->get_url(), + 'title' => _x( 'View', 'oss', 'woocommerce-germanized' ), + ), + 'export' => array( + 'url' => $report->get_export_link(), + 'title' => _x( 'Export', 'oss', 'woocommerce-germanized' ), + ), + 'export_bop' => array( + 'url' => $report->get_export_link( 'bop' ), + 'title' => _x( 'Export BOP', 'oss', 'woocommerce-germanized' ), + ), + 'refresh' => array( + 'url' => $report->get_refresh_link(), + 'title' => _x( 'Refresh', 'oss', 'woocommerce-germanized' ), + ), + 'delete' => array( + 'url' => $report->get_delete_link(), + 'title' => _x( 'Delete', 'oss', 'woocommerce-germanized' ), + ), + ); + + if ( 'completed' !== $report->get_status() ) { + $actions['cancel'] = $actions['delete']; + $actions['cancel']['title'] = _x( 'Cancel', 'oss', 'woocommerce-germanized' ); + + unset( $actions['view'] ); + unset( $actions['refresh'] ); + unset( $actions['delete'] ); + unset( $actions['export'] ); + unset( $actions['export_bop'] ); + } + + if ( 'DE' !== Helper::get_base_country() && isset( $actions['export_bop'] ) ) { + unset( $actions['export_bop'] ); + } + + if ( 'observer' === $report->get_type() ) { + unset( $actions['refresh'] ); + unset( $actions['cancel'] ); + } + + return $actions; + } + + public static function render_report_details() { + global $wp_list_table; + + $report_id = isset( $_GET['report'] ) ? wc_clean( wp_unslash( $_GET['report'] ) ) : false; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + if ( ! $report_id ) { + return; + } + + if ( ! $report = Package::get_report( $report_id ) ) { + return; + } + + $actions = self::get_report_actions( $report ); + unset( $actions['view'] ); + + $columns = array( + 'country' => _x( 'Country', 'oss', 'woocommerce-germanized' ), + 'tax_rate' => _x( 'Tax Rate', 'oss', 'woocommerce-germanized' ), + 'net_total' => _x( 'Net Total', 'oss', 'woocommerce-germanized' ), + 'tax_total' => _x( 'Tax Total', 'oss', 'woocommerce-germanized' ), + ); + + $countries = $report->get_countries(); + ?> +
+

get_title() ); ?>

+ + $action ) : ?> + + + + get_status() ) : ?> +

get_date_start()->date_i18n( wc_date_format() ) ); ?> – get_date_end()->date_i18n( wc_date_format() ) ); ?>: get_net_total() ); ?> (get_tax_total() ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>)

+
+ + + + + $column ) : ?> + + + + + + get_tax_rates_by_country( $country ) as $tax_rate ) : + ?> + + + + + + + + + +
get_country_net_total( $country, $tax_rate ) ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>get_country_tax_total( $country, $tax_rate ) ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>
+ + +

Find pending actions', 'oss', 'woocommerce-germanized' ), esc_html( $details['order_count'] ), ( $details['next_date'] ? esc_html( $details['next_date']->date_i18n( wc_date_format() . ' @ ' . wc_time_format() ) ) : esc_html_x( 'Not yet known', 'oss', 'woocommerce-germanized' ) ), esc_url( $details['link'] ) ); ?>

+ +
+ current_action(); + + if ( $doaction ) { + /** + * This nonce is dynamically constructed by WP_List_Table and uses + * the normalized plural argument. + */ + check_admin_referer( 'bulk-' . sanitize_key( _x( 'Reports', 'oss', 'woocommerce-germanized' ) ) ); + + $pagenum = $wp_list_table->get_pagenum(); + $parent_file = $wp_list_table->get_main_page(); + $sendback = remove_query_arg( array( 'deleted', 'ids', 'changed', 'bulk_action' ), wp_get_referer() ); + + if ( ! $sendback ) { + $sendback = admin_url( $parent_file ); + } + + $sendback = add_query_arg( 'paged', $pagenum, $sendback ); + $report_ids = array(); + + if ( isset( $_REQUEST['ids'] ) ) { + $report_ids = explode( ',', wc_clean( wp_unslash( $_REQUEST['ids'] ) ) ); + } elseif ( ! empty( $_REQUEST['report'] ) ) { + $report_ids = wc_clean( wp_unslash( $_REQUEST['report'] ) ); + } + + if ( ! empty( $report_ids ) ) { + $sendback = $wp_list_table->handle_bulk_actions( $doaction, $report_ids, $sendback ); + } + + $sendback = remove_query_arg( array( 'action', 'action2', '_status', 'bulk_edit', 'report', 'report_created' ), $sendback ); + + wp_safe_redirect( esc_url_raw( $sendback ) ); + exit(); + } elseif ( ! empty( $_REQUEST['_wp_http_referer'] ) ) { + wp_safe_redirect( esc_url_raw( remove_query_arg( array( '_wp_http_referer', '_wpnonce' ), esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) ) ) ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotValidated + exit; + } + + $wp_list_table->set_bulk_notice(); + $wp_list_table->prepare_items(); + + add_screen_option( 'per_page' ); + } + + public static function register_settings( $settings ) { + if ( ! Package::is_integration() ) { + $settings[] = new SettingsPage(); + } + + return $settings; + } + + public static function get_screen_ids() { + $screen_ids = array( 'woocommerce_page_wc-settings', 'woocommerce_page_oss-reports', 'product' ); + + return $screen_ids; + } + + public static function admin_styles() { + $screen = get_current_screen(); + $screen_id = $screen ? $screen->id : ''; + $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; + + wp_register_style( 'oss_woo', Package::get_url() . '/assets/css/admin' . $suffix . '.css', array(), Package::get_version() ); + + // Admin styles for WC pages only. + if ( in_array( $screen_id, self::get_screen_ids(), true ) ) { + wp_enqueue_style( 'oss_woo' ); + } + } + + public static function admin_scripts() { + global $post; + + $screen = get_current_screen(); + $screen_id = $screen ? $screen->id : ''; + $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; + $deps = array( 'jquery', 'woocommerce_admin' ); + + if ( in_array( $screen_id, array( 'woocommerce_page_oss-reports' ), true ) ) { + $deps[] = 'jquery-ui-datepicker'; + } + + wp_register_script( 'oss-admin', Package::get_assets_url() . '/js/admin' . $suffix . '.js', $deps, Package::get_version() ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.NotInFooter + + if ( in_array( $screen_id, self::get_screen_ids(), true ) ) { + wp_enqueue_script( 'oss-admin' ); + + wp_localize_script( + 'oss-admin', + 'oss_admin_params', + array( + 'ajax_url' => admin_url( 'admin-ajax.php' ), + ) + ); + } + } +} diff --git a/packages/one-stop-shop-woocommerce/src/AdminNote.php b/packages/one-stop-shop-woocommerce/src/AdminNote.php new file mode 100644 index 000000000..c5c6eef9d --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/AdminNote.php @@ -0,0 +1,92 @@ + 'oss_hide_notice', + 'notice' => static::get_id(), + '_wpnonce' => wp_create_nonce( 'oss_hide_notice' ), + ), + admin_url( 'admin-post.php' ) + ); + } + + public static function has_actions() { + $actions = static::get_actions(); + + return empty( $actions ) ? false : true; + } + + public static function get_actions() { + return array( + array( + 'target' => '', + 'title' => _x( 'Dismiss', 'oss', 'woocommerce-germanized' ), + 'url' => static::get_dismiss_url(), + 'is_primary' => false, + ), + ); + } + + public static function is_enabled() { + $enabled = true; + + if ( 'yes' === get_option( 'oss_hide_notice_' . sanitize_key( static::get_id() ) ) ) { + $enabled = false; + } + + return $enabled; + } + + public static function render() { + ?> +
+ + +

+ + + +

+ '', + 'url' => '', + 'is_primary' => true, + 'target' => '_blank', + ) + ); + ?> + + +

+ +
+ type = $type; + $default_end = new \WC_DateTime(); + $default_start = new \WC_DateTime( 'now' ); + $default_start->modify( '-1 year' ); + + $args = wp_parse_args( + $args, + array( + 'start' => $default_start->format( 'Y-m-d' ), + 'end' => $default_end->format( 'Y-m-d' ), + 'limit' => Queue::get_batch_size(), + 'status' => Queue::get_order_statuses(), + 'offset' => 0, + 'order_types' => array( 'shop_order' ), + 'orders_processed' => 0, + 'date_field' => Queue::use_date_paid() ? 'date_paid' : 'date_created', + ) + ); + + /** + * Observers do not treat refunds separately + */ + if ( 'observer' === $type ) { + $args['order_types'] = array( 'shop_order' ); + } + + foreach ( array( 'start', 'end' ) as $date_field ) { + if ( is_a( $args[ $date_field ], 'WC_DateTime' ) ) { + $args[ $date_field ] = $args[ $date_field ]->format( 'Y-m-d' ); + } elseif ( is_numeric( $args[ $date_field ] ) ) { + $date = new \WC_DateTime( '@' . $args[ $date_field ] ); + $args[ $date_field ] = $date->format( 'Y-m-d' ); + } + } + + $this->args = $args; + } + + public function get_type() { + return $this->type; + } + + public function get_args() { + return $this->args; + } + + public function get_id() { + return sanitize_key( 'oss_' . $this->type . '_report_' . $this->args['start'] . '_' . $this->args['end'] ); + } + + public function delete() { + $report = new Report( $this->get_id() ); + $report->delete(); + + delete_option( $this->get_id() . '_tmp_result' ); + } + + public function start() { + $report = new Report( $this->get_id() ); + $report->reset(); + $report->save(); + + return $report; + } + + /** + * @param \WC_Order $order + * + * @return mixed + */ + protected function get_order_taxable_country( $order ) { + if ( ! is_callable( array( $order, 'get_shipping_country' ) ) ) { + return Helper::get_base_country(); + } + + $taxable_country_type = ! empty( $order->get_shipping_country() ) ? 'shipping' : 'billing'; + $taxable_country = 'shipping' === $taxable_country_type ? $order->get_shipping_country() : $order->get_billing_country(); + + return $taxable_country; + } + + /** + * @param \WC_Order $order + * + * @return mixed + */ + protected function get_order_taxable_postcode( $order ) { + $taxable_type = ! empty( $order->get_shipping_postcode() ) ? 'shipping' : 'billing'; + $taxable_postcode = 'shipping' === $taxable_type ? $order->get_shipping_postcode() : $order->get_billing_postcode(); + + return $taxable_postcode; + } + + /** + * @param \WC_Order $order + * + * @return bool + */ + protected function include_order( $order ) { + $taxable_country = $this->get_order_taxable_country( $order ); + $taxable_postcode = $this->get_order_taxable_postcode( $order ); + $included = true; + + if ( ! Helper::is_eu_vat_country( $taxable_country, $taxable_postcode ) || Helper::get_base_country() === $taxable_country ) { + $included = false; + } + + if ( floatval( $order->get_total_tax() ) === 0.0 ) { + $included = false; + } + + return apply_filters( 'oss_woocommerce_report_include_order', $included, $order ); + } + + protected function get_taxable_country_iso( $country ) { + if ( 'GB' === $country ) { + $country = 'XI'; + } + + return $country; + } + + /** + * @param \WC_Order|\WC_Order_Refund $order + */ + protected function get_order_number( $order ) { + if ( is_callable( $order, 'get_order_number' ) ) { + return $order->get_order_number(); + } else { + return $order->get_id(); + } + } + + /** + * @return true|\WP_Error + */ + public function next() { + $args = $this->args; + $results = Queue::query( $args ); + $orders_processed = 0; + $tax_data = $this->get_temporary_result(); + $supports_refunds = in_array( 'shop_order_refund', $args['order_types'], true ); + + Package::extended_log( sprintf( '%d applicable orders found', count( $results ) ) ); + + if ( ! empty( $results ) ) { + foreach ( $results as $result ) { + if ( $order = wc_get_order( $result->ID ) ) { + $forced_parent_order = false; + + /** + * Query refund's parent order as the refund does not contain enough data (e.g. billing_country) + */ + if ( $order->get_parent_id() > 0 ) { + $forced_parent_order = wc_get_order( $order->get_parent_id() ); + + if ( ! $forced_parent_order ) { + continue; + } + + Package::extended_log( sprintf( 'Parent order: %s', $this->get_order_number( $forced_parent_order ) ) ); + } elseif ( is_callable( array( $order, 'get_shipping_country' ) ) ) { + $forced_parent_order = $order; + } + + if ( ! $forced_parent_order ) { + continue; + } + + $taxable_country = $this->get_order_taxable_country( $forced_parent_order ); + + if ( ! $this->include_order( $forced_parent_order ) ) { + Package::extended_log( sprintf( 'Skipping order #%1$s based on taxable country %2$s, tax total: %3$s', $this->get_order_number( $order ), $taxable_country, $order->get_total_tax() ) ); + continue; + } + + $country_iso = $this->get_taxable_country_iso( $taxable_country ); + + Package::extended_log( sprintf( 'Processing order #%1$s (%2$s) based on taxable country %3$s', $this->get_order_number( $order ), $order->get_type(), $country_iso ) ); + + if ( ! isset( $tax_data[ $country_iso ] ) ) { + $tax_data[ $country_iso ] = array(); + } + + foreach ( $order->get_taxes() as $key => $tax ) { + $tax_percent = (float) Helper::get_tax_rate_percent( $tax->get_rate_id(), $forced_parent_order ); + $tax_total = (float) $tax->get_tax_total() + (float) $tax->get_shipping_tax_total(); + + /** + * Do only remove refunded tax total in case this query does not explicitly support refunds (e.g. observers) + */ + if ( ! $supports_refunds ) { + $refunded = (float) $forced_parent_order->get_total_tax_refunded_by_rate_id( $tax->get_rate_id() ); + $tax_total = $tax_total - $refunded; + + Package::extended_log( sprintf( 'Refunded tax %1$s = %2$s', $tax_percent, $refunded ) ); + } + + if ( $tax_percent <= 0 || 0.0 === $tax_total ) { + if ( $tax_percent <= 0 ) { + Package::extended_log( sprintf( 'Skipping order due to missing tax percentage' ) ); + } + + if ( 0.0 === $tax_total ) { + Package::extended_log( sprintf( 'Skipping order due to tax total = 0' ) ); + } + + continue; + } + + if ( ! isset( $tax_data[ $country_iso ][ "$tax_percent" ] ) ) { + $tax_data[ $country_iso ][ "$tax_percent" ] = array( + 'tax_total' => 0, + 'net_total' => 0, + ); + } + + $net_total = ( $tax_total / ( (float) $tax_percent / 100 ) ); + + Package::extended_log( sprintf( 'Tax total %1$s = %2$s', $tax_percent, $tax_total ) ); + Package::extended_log( sprintf( 'Net total %1$s = %2$s', $tax_percent, $net_total ) ); + + $net_total = wc_add_number_precision( $net_total, false ); + $tax_total = wc_add_number_precision( $tax_total, false ); + + $tax_data[ $country_iso ][ "$tax_percent" ]['tax_total'] = (float) $tax_data[ $country_iso ][ "$tax_percent" ]['tax_total']; + $tax_data[ $country_iso ][ "$tax_percent" ]['tax_total'] += $tax_total; + + $tax_data[ $country_iso ][ "$tax_percent" ]['net_total'] = (float) $tax_data[ $country_iso ][ "$tax_percent" ]['net_total']; + $tax_data[ $country_iso ][ "$tax_percent" ]['net_total'] += $net_total; + + $orders_processed++; + } + } + } + + $this->args['orders_processed'] = absint( $this->args['orders_processed'] ) + $orders_processed; + + update_option( $this->get_id() . '_tmp_result', $tax_data, false ); + + return true; + } else { + return new \WP_Error( 'empty', _x( 'No orders found.', 'oss', 'woocommerce-germanized' ) ); + } + } + + /** + * @return Report + */ + public function complete() { + Package::extended_log( sprintf( 'Completed called' ) ); + + $tmp_result = $this->get_temporary_result(); + $report = new Report( $this->get_id() ); + $tax_total = 0; + $net_total = 0; + + foreach ( $tmp_result as $country => $tax_data ) { + foreach ( $tax_data as $percent => $totals ) { + $tax_total += (float) $totals['tax_total']; + $net_total += (float) $totals['net_total']; + + $report->set_country_net_total( $country, $percent, (float) wc_remove_number_precision( $totals['net_total'] ) ); + $report->set_country_tax_total( $country, $percent, (float) wc_remove_number_precision( $totals['tax_total'] ) ); + } + } + + $net_total = (float) wc_remove_number_precision( $net_total ); + $tax_total = (float) wc_remove_number_precision( $tax_total ); + + Package::extended_log( sprintf( 'Completed net total: %s', $net_total ) ); + Package::extended_log( sprintf( 'Completed tax total: %s', $tax_total ) ); + + $report->set_net_total( $net_total ); + $report->set_tax_total( $tax_total ); + $report->set_status( 'completed' ); + $report->set_version( Package::get_version() ); + $report->save(); + + return $report; + } + + protected function get_temporary_result() { + return (array) get_option( $this->get_id() . '_tmp_result', array() ); + } +} diff --git a/packages/one-stop-shop-woocommerce/src/CSVExporter.php b/packages/one-stop-shop-woocommerce/src/CSVExporter.php new file mode 100644 index 000000000..89ffa40f8 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/CSVExporter.php @@ -0,0 +1,119 @@ +report = new Report( $id ); + $this->decimals = apply_filters( 'oss_woocommerce_csv_export_decimals', $decimals, $this ); + $this->column_names = $this->get_default_column_names(); + $this->filename = sanitize_file_name( $this->report->get_id() . '.csv' ); + } + + /** + * Return an array of columns to export. + * + * @since 3.1.0 + * @return array + */ + public function get_default_column_names() { + return apply_filters( + 'one_stop_shop_woocommerce_export_default_columns', + array( + 'country' => _x( 'Country code', 'oss', 'woocommerce-germanized' ), + 'tax_rate' => _x( 'Tax rate', 'oss', 'woocommerce-germanized' ), + 'taxable_base' => _x( 'Taxable base', 'oss', 'woocommerce-germanized' ), + 'amount' => _x( 'Amount', 'oss', 'woocommerce-germanized' ), + ) + ); + } + + public function get_report() { + return $this->report; + } + + public function get_decimals() { + return $this->decimals; + } + + protected function format_decimal( $value ) { + return wc_format_decimal( $value, $this->get_decimals() ); + } + + protected function format_country( $country ) { + return strtoupper( $country ); + } + + protected function get_row_data( $country, $tax_rate ) { + $row = array(); + + foreach ( array_keys( $this->get_column_names() ) as $column_id ) { + $column_id = strstr( $column_id, ':' ) ? current( explode( ':', $column_id ) ) : $column_id; + $value = ''; + + if ( 'country' === $column_id ) { + $value = $this->format_country( $country ); + } elseif ( 'tax_rate' === $column_id ) { + $value = $this->format_decimal( $tax_rate ); + } elseif ( 'taxable_base' === $column_id ) { + $value = $this->format_decimal( $this->report->get_country_net_total( $country, $tax_rate, $this->get_decimals() ) ); + } elseif ( 'amount' === $column_id ) { + $value = $this->format_decimal( $this->report->get_country_tax_total( $country, $tax_rate, $this->get_decimals() ) ); + } elseif ( is_callable( array( $this, "get_column_value_{$column_id}" ) ) ) { + $value = $this->{"get_column_value_{$column_id}"}( $country, $tax_rate ); + } else { + $value = apply_filters( "one_stop_shop_woocommerce_export_column_{$column_id}", $value, $country, $tax_rate, $this ); + } + + $row[ $column_id ] = $value; + } + + return $row; + } + + /** + * Prepare data that will be exported. + */ + public function prepare_data_to_export() { + $countries = $this->report->get_countries(); + + if ( ! empty( $countries ) ) { + foreach ( $countries as $country ) { + foreach ( $this->report->get_tax_rates_by_country( $country ) as $tax_rate ) { + $this->row_data[] = apply_filters( 'one_stop_shop_woocommerce_export_row_data', $this->get_row_data( $country, $tax_rate ), $country, $tax_rate, $this ); + } + } + } + } +} diff --git a/packages/one-stop-shop-woocommerce/src/CSVExporterBOP.php b/packages/one-stop-shop-woocommerce/src/CSVExporterBOP.php new file mode 100644 index 000000000..f832f942b --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/CSVExporterBOP.php @@ -0,0 +1,125 @@ + 'Satzart', + 'country' => 'Land des Verbrauchs', + 'tax_type' => 'Umsatzsteuertyp', + 'tax_rate' => 'Umsatzsteuersatz', + 'taxable_base' => 'Steuerbemessungsgrundlage, Nettobetrag', + 'amount' => 'Umsatzsteuerbetrag', + ) + ); + } + + protected function get_column_value_bop_type( $country, $tax_rate ) { + return apply_filters( 'one_stop_shop_woocommerce_bop_export_type', 3 ); + } + + protected function get_column_value_tax_type( $country, $tax_rate ) { + $tax_type = Helper::get_tax_type_by_country_rate( $tax_rate, $country ); + $tax_return_type = 'STANDARD'; + + switch ( $tax_type ) { + case 'reduced': + case 'greater-reduced': + case 'super-reduced': + $tax_return_type = 'REDUCED'; + break; + default: + $tax_return_type = strtoupper( $tax_type ); + break; + } + + return $tax_return_type; + } + + protected function format_country( $country ) { + $country = parent::format_country( $country ); + + if ( 'GR' === $country ) { + $country = 'EL'; + } + + return $country; + } + + /** + * Prepare data that will be exported. + */ + public function prepare_data_to_export() { + $countries = $this->report->get_countries(); + + if ( ! empty( $countries ) ) { + foreach ( $countries as $country ) { + $tax_rates = $this->report->get_tax_rates_by_country( $country ); + + if ( ! empty( $tax_rates ) ) { + $this->row_data[] = apply_filters( + 'one_stop_shop_woocommerce_export_bop_country_header_data', + array( + 'country' => $this->format_country( $country ), + 'bop_type' => 1, + ), + $country, + $this + ); + + foreach ( $tax_rates as $tax_rate ) { + $this->row_data[] = apply_filters( 'one_stop_shop_woocommerce_bop_export_row_data', $this->get_row_data( $country, $tax_rate ), $country, $tax_rate, $this ); + } + } + } + } + } + + /** + * Do the export. Prevent Woo from prepending a BOM. + */ + public function export() { + $this->prepare_data_to_export(); + $this->send_headers(); + + $csv_data = $this->export_column_headers() . $this->get_csv_data(); + + // Replace newlines with Windows-style. + $csv_data = preg_replace( '~\R~u', "\r", $csv_data ); + + $this->send_content( $csv_data ); + die(); + } + + protected function export_column_headers() { + $buffer = fopen( 'php://output', 'w' ); // phpcs:ignore WordPress.WP.AlternativeFunctions.file_system_read_fopen + ob_start(); + fwrite( $buffer, '#v1.1' . PHP_EOL ); // phpcs:ignore WordPress.WP.AlternativeFunctions.file_system_read_fwrite + $content = ob_get_clean(); + + return $content; + } + + protected function fputcsv( $buffer, $export_row ) { + fputcsv( $buffer, $export_row, $this->get_delimiter(), "'", "\0" ); // @codingStandardsIgnoreLine + } +} diff --git a/packages/one-stop-shop-woocommerce/src/DeliveryThresholdEmailNotification.php b/packages/one-stop-shop-woocommerce/src/DeliveryThresholdEmailNotification.php new file mode 100644 index 000000000..39d7ffc80 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/DeliveryThresholdEmailNotification.php @@ -0,0 +1,113 @@ +template_base = Package::get_path() . '/templates/'; + $this->id = 'oss_delivery_threshold_email_notification'; + $this->title = _x( 'OSS Delivery Threshold Notification', 'oss', 'woocommerce-germanized' ); + $this->description = _x( 'This email notifies shop owners in case the delivery threshold (OSS) is close to being reached.', 'oss', 'woocommerce-germanized' ); + $this->template_html = 'emails/admin-delivery-threshold.php'; + $this->template_plain = 'emails/plain/admin-delivery-threshold.php'; + $this->customer_email = false; + + parent::__construct(); + + // Other settings. + $this->recipient = $this->get_option( 'recipient', get_option( 'admin_email' ) ); + } + + /** + * Get email subject. + * + * @since 3.1.0 + * @return string + */ + public function get_default_subject() { + return _x( '[{site_title}]: OSS delivery threshold reached', 'oss', 'woocommerce-germanized' ); + } + + /** + * Get email heading. + * + * @since 3.1.0 + * @return string + */ + public function get_default_heading() { + return _x( 'OSS delivery threshold reached', 'oss', 'woocommerce-germanized' ); + } + + /** + * Get content html. + * + * @return string + */ + public function get_content_html() { + return wc_get_template_html( + $this->template_html, + array( + 'report' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => true, + 'plain_text' => false, + 'email' => $this, + ), + '', + $this->template_base + ); + } + + /** + * Get content plain. + * + * @return string + */ + public function get_content_plain() { + return wc_get_template_html( + $this->template_plain, + array( + 'report' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => true, + 'plain_text' => true, + 'email' => $this, + ), + '', + $this->template_base + ); + } + + /** + * Trigger the sending of this email. + * + * @param Report $report + */ + public function trigger( $report ) { + $this->object = $report; + + $success = $this->send( + $this->get_recipient(), + $this->get_subject(), + $this->get_content(), + $this->get_headers(), + $this->get_attachments() + ); + + if ( $success ) { + update_option( 'oss_woocommerce_notification_sent_' . $report->get_date_start()->format( 'Y' ), 'yes', false ); + } + } +} diff --git a/packages/one-stop-shop-woocommerce/src/DeliveryThresholdWarning.php b/packages/one-stop-shop-woocommerce/src/DeliveryThresholdWarning.php new file mode 100644 index 000000000..59d023aa8 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/DeliveryThresholdWarning.php @@ -0,0 +1,40 @@ + '', + 'title' => _x( 'See details', 'oss', 'woocommerce-germanized' ), + 'url' => Settings::get_settings_url(), + 'is_primary' => true, + ), + ), + parent::get_actions() + ); + } + + public static function get_content() { + return Admin::get_threshold_notice_content(); + } + + public static function get_title() { + return Admin::get_threshold_notice_title(); + } + + public static function is_enabled() { + $is_enabled = parent::is_enabled(); + + return $is_enabled && Package::enable_auto_observer() && Package::observer_report_needs_notification(); + } + + public static function get_id() { + return 'delivery-threshold-warning-' . date( 'Y' ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + } +} diff --git a/packages/one-stop-shop-woocommerce/src/Install.php b/packages/one-stop-shop-woocommerce/src/Install.php new file mode 100644 index 000000000..1aae1b7c3 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/Install.php @@ -0,0 +1,41 @@ +get_status() ) { + $report->delete(); + } + } + } + } + + /** + * Make sure there is only one observer running at a time. + */ + foreach ( $running as $k => $report_id ) { + if ( in_array( $report_id, $running_observers, true ) && $report_id !== $has_running_observer ) { + if ( $report = self::get_report( $report_id ) ) { + $report->delete(); + } + + unset( $running[ $k ] ); + } + } + + $running = array_values( $running ); + + update_option( 'oss_woocommerce_reports_running', $running, false ); + Queue::clear_cache(); + + $observer_reports = self::get_reports( + array( + 'type' => 'observer', + 'include_observer' => true, + ) + ); + + foreach ( $observer_reports as $observer ) { + if ( ! self::enable_auto_observer() ) { + /** + * Delete observers in case observing was disabled. + */ + $observer->delete(); + } else { + /* + * Do not delete running observers (which are orphans by design) + */ + if ( $observer->get_id() === $has_running_observer ) { + continue; + } + + $year = $observer->get_date_start()->format( 'Y' ); + + /** + * Delete orphan observer reports (reports not linked as a main observer for a certain year). + */ + if ( get_option( 'oss_woocommerce_observer_report_' . $year ) !== $observer->get_id() ) { + $observer->delete(); + } + } + } + + /** + * In case the current observer report does not exist - delete the option + */ + if ( self::enable_auto_observer() ) { + $year = date( 'Y' ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + $report_id = get_option( 'oss_woocommerce_observer_report_' . $year ); + + if ( ! empty( $report_id ) ) { + if ( ! self::get_report( $report_id ) ) { + delete_option( 'oss_woocommerce_observer_report_' . $year ); + } + } + } + } + + public static function dependency_notice() { + ?> +

+ get_net_total(); + } + + $total_left = self::get_delivery_threshold() - $net_total; + + if ( $total_left <= 0 ) { + $total_left = 0; + } + + return $total_left; + } + + /** + * @param null $year + * + * @return false|Report + */ + public static function get_completed_observer_report( $year = null ) { + $observer_report = self::get_observer_report( $year ); + + if ( ! $observer_report || 'completed' !== $observer_report->get_status() ) { + return false; + } + + return $observer_report; + } + + /** + * @param null $year + * + * @return false|Report + */ + public static function get_observer_report( $year = null ) { + if ( is_null( $year ) ) { + $year = date( 'Y' ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + } + + $report_id = get_option( 'oss_woocommerce_observer_report_' . $year ); + $report = false; + + if ( ! empty( $report_id ) ) { + $report = self::get_report( $report_id ); + } + + return $report; + } + + public static function observer_report_is_outdated() { + $is_outdated = true; + + if ( $observer = self::get_observer_report() ) { + $date_end = $observer->get_date_end(); + $now = new \WC_DateTime(); + + $diff = $now->diff( $date_end ); + + if ( $diff->days <= 1 ) { + $is_outdated = false; + } + } + + return $is_outdated; + } + + public static function string_to_datetime( $time_string ) { + if ( is_string( $time_string ) && ! is_numeric( $time_string ) ) { + $time_string = strtotime( $time_string ); + } + + $date_time = $time_string; + + if ( is_numeric( $date_time ) ) { + $date_time = new \WC_DateTime( "@{$date_time}", new \DateTimeZone( 'UTC' ) ); + } + + if ( ! is_a( $date_time, 'WC_DateTime' ) ) { + return null; + } + + return $date_time; + } + + /** + * @param $id + * + * @return false|Report + */ + public static function get_report( $id ) { + $report = new Report( $id ); + + if ( $report->exists() ) { + return $report; + } + + return false; + } + + public static function get_report_id( $parts ) { + $parts = wp_parse_args( + $parts, + array( + 'type' => 'daily', + 'date_start' => date( 'Y-m-d' ), // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + 'date_end' => date( 'Y-m-d' ), // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + ) + ); + + if ( is_a( $parts['date_start'], 'WC_DateTime' ) ) { + $parts['date_start'] = $parts['date_start']->format( 'Y-m-d' ); + } + + if ( is_a( $parts['date_end'], 'WC_DateTime' ) ) { + $parts['date_end'] = $parts['date_end']->format( 'Y-m-d' ); + } + + return 'oss_' . $parts['type'] . '_report_' . $parts['date_start'] . '_' . $parts['date_end']; + } + + public static function get_report_data( $id ) { + $id_parts = explode( '_', $id ); + $data = array( + 'id' => $id, + 'type' => $id_parts[1], + 'date_start' => self::string_to_datetime( $id_parts[3] ), + 'date_end' => self::string_to_datetime( $id_parts[4] ), + ); + + return $data; + } + + public static function get_report_title( $id ) { + $args = self::get_report_data( $id ); + $title = _x( 'Report', 'oss', 'woocommerce-germanized' ); + + if ( 'quarterly' === $args['type'] ) { + $date_start = $args['date_start']; + $quarter = 1; + $month_num = (int) $date_start->date_i18n( 'n' ); + + if ( 4 === $month_num ) { + $quarter = 2; + } elseif ( 7 === $month_num ) { + $quarter = 3; + } elseif ( 10 === $month_num ) { + $quarter = 4; + } + + $title = sprintf( _x( 'Q%1$s/%2$s', 'oss', 'woocommerce-germanized' ), $quarter, $date_start->date_i18n( 'Y' ) ); + } elseif ( 'monthly' === $args['type'] ) { + $date_start = $args['date_start']; + $month_num = $date_start->date_i18n( 'm' ); + + $title = sprintf( _x( '%1$s/%2$s', 'oss', 'woocommerce-germanized' ), $month_num, $date_start->date_i18n( 'Y' ) ); + } elseif ( 'yearly' === $args['type'] ) { + $date_start = $args['date_start']; + + $title = sprintf( _x( '%1$s', 'oss', 'woocommerce-germanized' ), $date_start->date_i18n( 'Y' ) ); // phpcs:ignore WordPress.WP.I18n.NoEmptyStrings + } elseif ( 'custom' === $args['type'] ) { + $date_start = $args['date_start']; + $date_end = $args['date_end']; + + $title = sprintf( _x( '%1$s - %2$s', 'oss', 'woocommerce-germanized' ), $date_start->date_i18n( 'Y-m-d' ), $date_end->date_i18n( 'Y-m-d' ) ); + } elseif ( 'observer' === $args['type'] ) { + $date_start = $args['date_start']; + $date_end = $args['date_end']; + + $title = sprintf( _x( 'Observer %1$s', 'oss', 'woocommerce-germanized' ), $date_start->date_i18n( 'Y' ) ); + } + + return $title; + } + + /** + * @param Report $report + */ + public static function remove_report( $report ) { + $reports_available = self::get_report_ids(); + + if ( in_array( $report->get_id(), $reports_available[ $report->get_type() ], true ) ) { + $reports_available[ $report->get_type() ] = array_diff( $reports_available[ $report->get_type() ], array( $report->get_id() ) ); + + update_option( 'oss_woocommerce_reports', $reports_available, false ); + + /** + * Force non-cached option + */ + wp_cache_delete( 'oss_woocommerce_reports', 'options' ); + } + } + + /** + * @param array $args + * + * @return Report[] + */ + public static function get_reports( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'type' => '', + 'limit' => -1, + 'offset' => 0, + 'orderby' => 'date_start', + 'include_observer' => false, + ) + ); + + $ids = self::get_report_ids( $args['include_observer'] ); + + if ( ! empty( $args['type'] ) ) { + $report_ids = array_key_exists( $args['type'], $ids ) ? $ids[ $args['type'] ] : array(); + } else { + $report_ids = array_merge( ...array_values( $ids ) ); + } + + $reports_sorted = array(); + + foreach ( $report_ids as $id ) { + $reports_sorted[] = self::get_report_data( $id ); + } + + if ( array_key_exists( $args['orderby'], array( 'date_start', 'date_end' ) ) ) { + usort( + $reports_sorted, + function( $a, $b ) use ( $args ) { + if ( $a[ $args['orderby'] ] === $b[ $args['orderby'] ] ) { + return 0; + } + + return $a[ $args['orderby'] ] < $b[ $args['orderby'] ] ? -1 : 1; + } + ); + } + + if ( -1 !== $args['limit'] ) { + $reports_sorted = array_slice( $reports_sorted, $args['offset'], $args['limit'] ); + } + + $reports = array(); + + foreach ( $reports_sorted as $data ) { + if ( $report = self::get_report( $data['id'] ) ) { + $reports[] = $report; + } + } + + return $reports; + } + + public static function clear_caches() { + delete_transient( 'oss_reports_counts' ); + wp_cache_delete( 'oss_woocommerce_reports', 'options' ); + } + + public static function get_report_counts() { + $types = array_keys( self::get_available_report_types( true ) ); + $cache_key = 'oss_reports_counts'; + $counts = get_transient( $cache_key ); + + if ( false === $counts ) { + $counts = array(); + + foreach ( $types as $type ) { + $counts[ $type ] = 0; + } + + foreach ( self::get_reports( array( 'include_observer' => true ) ) as $report ) { + if ( ! array_key_exists( $report->get_type(), $counts ) ) { + continue; + } + + $counts[ $report->get_type() ] += 1; + } + + set_transient( $cache_key, $counts ); + } + + return (array) $counts; + } + + public static function load_plugin_textdomain() { + add_filter( 'plugin_locale', array( __CLASS__, 'support_german_language_variants' ), 10, 2 ); + + if ( function_exists( 'determine_locale' ) ) { + $locale = determine_locale(); + } else { + // @todo Remove when start supporting WP 5.0 or later. + $locale = is_admin() ? get_user_locale() : get_locale(); + } + + $locale = apply_filters( 'plugin_locale', $locale, 'woocommerce-germanized' ); + + unload_textdomain( 'one-stop-shop-woocommerce' ); + load_textdomain( 'one-stop-shop-woocommerce', trailingslashit( WP_LANG_DIR ) . 'one-stop-shop-woocommerce/one-stop-shop-woocommerce-' . $locale . '.mo' ); + load_plugin_textdomain( 'one-stop-shop-woocommerce', false, plugin_basename( self::get_path() ) . '/i18n/languages/' ); + } + + public static function support_german_language_variants( $locale, $domain ) { + if ( 'one-stop-shop-woocommerce' === $domain && apply_filters( 'oss_woocommerce_force_de_language', in_array( $locale, array( 'de_CH', 'de_AT' ), true ) ) ) { + $locale = 'de_DE'; + } + + return $locale; + } + + public static function register_emails( $emails ) { + $mails = array( + '\Vendidero\OneStopShop\DeliveryThresholdEmailNotification', + ); + + foreach ( $mails as $mail ) { + $emails[ self::sanitize_email_class( $mail ) ] = new $mail(); + } + + return $emails; + } + + protected static function sanitize_email_class( $class ) { + return 'oss_woocommerce_' . sanitize_key( str_replace( __NAMESPACE__ . '\\', '', $class ) ); + } + + public static function observer_report_needs_notification() { + $needs_notification = false; + + if ( $report = self::get_observer_report() ) { + $net_total = $report->get_net_total(); + $threshold = self::get_delivery_notification_threshold(); + + if ( $net_total >= $threshold ) { + $needs_notification = true; + } + } + + return apply_filters( 'oss_woocommerce_observer_report_needs_notification', $needs_notification ); + } + + /** + * @param Report $observer_report + */ + public static function maybe_send_notification( $observer_report ) { + if ( self::observer_report_needs_notification() ) { + if ( 'yes' !== get_option( 'oss_woocommerce_notification_sent_' . $observer_report->get_date_start()->format( 'Y' ) ) ) { + $mails = WC()->mailer()->get_emails(); + $mail = self::sanitize_email_class( '\Vendidero\OneStopShop\DeliveryThresholdEmailNotification' ); + + if ( isset( $mails[ $mail ] ) ) { + $mails[ $mail ]->trigger( $observer_report ); + } + } + } + } + + /** + * Let the observer date back 7 days to make sure most of the orders + * have already been processed (e.g. received payment etc) to reduce the chance of missing out on orders. + * + * @return int + */ + public static function get_observer_backdating_days() { + return 7; + } + + public static function update_observer_report() { + if ( self::enable_auto_observer() ) { + /** + * Delete observer reports with missing versions to make sure the report + * is re-created with the new backdating functionality. + */ + if ( $report = self::get_observer_report() ) { + if ( '' === $report->get_version() ) { + $report->delete(); + } + } + + $days = (int) self::get_observer_backdating_days(); + + $date_start = new \WC_DateTime(); + $date_start->modify( "-{$days} day" . ( $days > 1 ? 's' : '' ) ); + + Queue::start( 'observer', $date_start ); + } + } + + public static function setup_recurring_actions() { + if ( $queue = Queue::get_queue() ) { + + // Schedule once per day at 2:00 + if ( null === $queue->get_next( 'oss_woocommerce_daily_cleanup', array(), 'oss_woocommerce' ) ) { + $timestamp = strtotime( 'tomorrow midnight' ); + $date = new \WC_DateTime(); + + $date->setTimestamp( $timestamp ); + $date->modify( '+2 hours' ); + + $queue->cancel_all( 'oss_woocommerce_daily_cleanup', array(), 'oss_woocommerce' ); + $queue->schedule_recurring( $date->getTimestamp(), DAY_IN_SECONDS, 'oss_woocommerce_daily_cleanup', array(), 'oss_woocommerce' ); + } + + if ( self::enable_auto_observer() ) { + // Schedule once per day at 3:00 + if ( null === $queue->get_next( 'oss_woocommerce_daily_observer', array(), 'oss_woocommerce' ) ) { + $timestamp = strtotime( 'tomorrow midnight' ); + $date = new \WC_DateTime(); + + $date->setTimestamp( $timestamp ); + $date->modify( '+3 hours' ); + + $queue->cancel_all( 'oss_woocommerce_daily_observer', array(), 'oss_woocommerce' ); + $queue->schedule_recurring( $date->getTimestamp(), DAY_IN_SECONDS, 'oss_woocommerce_daily_observer', array(), 'oss_woocommerce' ); + } + } else { + $queue->cancel( 'oss_woocommerce_daily_observer', array(), 'oss_woocommerce' ); + } + } + } + + public static function get_available_report_types( $include_observer = false ) { + $types = array( + 'quarterly' => _x( 'Quarterly', 'oss', 'woocommerce-germanized' ), + 'yearly' => _x( 'Yearly', 'oss', 'woocommerce-germanized' ), + 'monthly' => _x( 'Monthly', 'oss', 'woocommerce-germanized' ), + 'custom' => _x( 'Custom', 'oss', 'woocommerce-germanized' ), + ); + + if ( $include_observer ) { + $types['observer'] = _x( 'Observer', 'oss', 'woocommerce-germanized' ); + } + + return $types; + } + + public static function get_type_title( $type ) { + $types = self::get_available_report_types( true ); + + return array_key_exists( $type, $types ) ? $types[ $type ] : ''; + } + + public static function get_report_statuses() { + return array( + 'pending' => _x( 'Pending', 'oss', 'woocommerce-germanized' ), + 'completed' => _x( 'Completed', 'oss', 'woocommerce-germanized' ), + 'failed' => _x( 'Failed', 'oss', 'woocommerce-germanized' ), + ); + } + + public static function get_report_status_title( $status ) { + $statuses = self::get_report_statuses(); + + return array_key_exists( $status, $statuses ) ? $statuses[ $status ] : ''; + } + + public static function has_dependencies() { + return ( class_exists( 'WooCommerce' ) ); + } + + public static function install() { + self::init(); + Install::install(); + } + + public static function deactivate() { + if ( self::has_dependencies() && Admin::supports_wc_admin() ) { + foreach ( Admin::get_notes() as $oss_note ) { + Admin::delete_wc_admin_note( $oss_note ); + } + } + } + + public static function install_integration() { + self::install(); + } + + public static function is_integration() { + $gzd_installed = class_exists( 'WooCommerce_Germanized' ); + $gzd_version = get_option( 'woocommerce_gzd_version', '1.0' ); + + return $gzd_installed && version_compare( $gzd_version, '3.5.0', '>=' ) ? true : false; + } + + /** + * Return the version of the package. + * + * @return string + */ + public static function get_version() { + return self::VERSION; + } + + /** + * Return the path to the package. + * + * @return string + */ + public static function get_path() { + return dirname( __DIR__ ); + } + + /** + * Return the path to the package. + * + * @return string + */ + public static function get_url() { + return plugins_url( '', __DIR__ ); + } + + public static function get_assets_url() { + return self::get_url() . '/assets'; + } + + private static function define_constant( $name, $value ) { + if ( ! defined( $name ) ) { + define( $name, $value ); + } + } + + public static function log( $message, $type = 'info' ) { + $logger = wc_get_logger(); + + if ( ! $logger || ! apply_filters( 'oss_woocommerce_enable_logging', true ) ) { + return; + } + + if ( ! is_callable( array( $logger, $type ) ) ) { + $type = 'info'; + } + + $logger->{$type}( $message, array( 'source' => 'one-stop-shop-woocommerce' ) ); + } + + public static function extended_log( $message, $type = 'info' ) { + if ( apply_filters( 'oss_woocommerce_enable_extended_logging', ( defined( 'WP_DEBUG' ) && WP_DEBUG ) ) ) { + self::log( $message, $type ); + } + } +} diff --git a/packages/one-stop-shop-woocommerce/src/Queue.php b/packages/one-stop-shop-woocommerce/src/Queue.php new file mode 100644 index 000000000..bbec8efbb --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/Queue.php @@ -0,0 +1,514 @@ +diff( $args['end'] ); + + /** + * Except observers, all new queries treat refunds separately + */ + if ( 'observer' !== $type ) { + $args['order_types'] = array( + 'shop_order', + 'shop_order_refund', + ); + } + + // Add version + $args['version'] = Package::get_version(); + + $generator = new AsyncReportGenerator( $type, $args ); + $queue_args = $generator->get_args(); + $queue = self::get_queue(); + + self::cancel( $generator->get_id() ); + + $report = $generator->start(); + + if ( is_a( $report, '\Vendidero\OneStopShop\Report' ) && $report->exists() ) { + Package::log( sprintf( 'Starting new %1$s', $report->get_title() ) ); + Package::extended_log( sprintf( 'Default report arguments: %s', wc_print_r( $queue_args, true ) ) ); + + $queue->schedule_single( + time() + 10, + 'oss_woocommerce_' . $generator->get_id(), + array( 'args' => $queue_args ), + 'oss_woocommerce' + ); + + $running = self::get_reports_running(); + + if ( ! in_array( $generator->get_id(), $running, true ) ) { + $running[] = $generator->get_id(); + } + + update_option( 'oss_woocommerce_reports_running', $running, false ); + self::clear_cache(); + + return $generator->get_id(); + } + + return false; + } + + public static function clear_cache() { + wp_cache_delete( 'oss_woocommerce_reports_running', 'options' ); + } + + public static function get_queue_details( $report_id ) { + $details = array( + 'next_date' => null, + 'link' => admin_url( 'admin.php?page=wc-status&tab=action-scheduler&s=' . esc_attr( $report_id ) . '&status=pending' ), + 'order_count' => 0, + 'has_action' => false, + 'is_finished' => false, + 'action' => false, + ); + + if ( $queue = self::get_queue() ) { + + if ( $next_date = $queue->get_next( 'oss_woocommerce_' . $report_id ) ) { + $details['next_date'] = $next_date; + } + + $search_args = array( + 'hook' => 'oss_woocommerce_' . $report_id, + 'status' => \ActionScheduler_Store::STATUS_RUNNING, + 'order' => 'DESC', + 'per_page' => 1, + ); + + $results = $queue->search( $search_args ); + + /** + * Search for pending as fallback + */ + if ( empty( $results ) ) { + $search_args['status'] = \ActionScheduler_Store::STATUS_PENDING; + $results = $queue->search( $search_args ); + } + + /** + * Last resort: Search for completed (e.g. if no pending and no running are found - must have been completed) + */ + if ( empty( $results ) ) { + $search_args['status'] = \ActionScheduler_Store::STATUS_COMPLETE; + $results = $queue->search( $search_args ); + } + + if ( ! empty( $results ) ) { + $action = array_values( $results )[0]; + $args = $action->get_args(); + $processed = isset( $args['args']['orders_processed'] ) ? (int) $args['args']['orders_processed'] : 0; + + $details['order_count'] = absint( $processed ); + $details['has_action'] = true; + $details['action'] = $action; + $details['is_finished'] = $action->is_finished(); + } + } + + return $details; + } + + public static function get_batch_size() { + return apply_filters( 'oss_woocommerce_report_batch_size', 25 ); + } + + public static function use_date_paid() { + $use_date_paid = 'date_paid' === get_option( 'oss_report_date_type', 'date_paid' ); + + return apply_filters( 'oss_woocommerce_report_use_date_paid', $use_date_paid ); + } + + public static function get_order_statuses() { + $statuses = array_keys( wc_get_order_statuses() ); + $statuses = array_diff( $statuses, array( 'wc-cancelled', 'wc-failed' ) ); + + if ( self::use_date_paid() ) { + $statuses = array_diff( $statuses, array( 'wc-pending' ) ); + } + + return apply_filters( 'oss_woocommerce_valid_order_statuses', $statuses ); + } + + public static function build_query( $args ) { + global $wpdb; + + $joins = array( + "LEFT JOIN {$wpdb->postmeta} AS mt1 ON {$wpdb->posts}.ID = mt1.post_id AND (mt1.meta_key = '_shipping_country' OR mt1.meta_key = '_billing_country')", + ); + + $taxable_countries_in = self::generate_in_query_sql( Helper::get_non_base_eu_countries( true ) ); + $post_status_in = self::generate_in_query_sql( $args['status'] ); + $post_type_in = self::generate_in_query_sql( isset( $args['order_types'] ) ? (array) $args['order_types'] : array( 'shop_order' ) ); + $where_country_sql = "mt1.meta_value IN {$taxable_countries_in}"; + + if ( in_array( 'shop_order_refund', $args['order_types'], true ) ) { + $joins[] = "LEFT JOIN {$wpdb->postmeta} AS mt1_parent ON {$wpdb->posts}.post_parent = mt1_parent.post_id AND (mt1_parent.meta_key = '_shipping_country' OR mt1_parent.meta_key = '_billing_country')"; + $where_country_sql = "( {$wpdb->posts}.post_parent > 0 AND (mt1_parent.meta_value IN {$taxable_countries_in}) ) OR ( mt1.meta_value IN {$taxable_countries_in} )"; + } + + $where_date_sql = $wpdb->prepare( "{$wpdb->posts}.post_date >= %s AND {$wpdb->posts}.post_date <= %s", $args['start'], $args['end'] ); + + if ( 'date_paid' === $args['date_field'] ) { + /** + * Add one day to the end date to capture timestamps (including time data) in between + */ + $end_adjusted = strtotime( $args['end'] ) + DAY_IN_SECONDS; + + /** + * Use a max end date to limit potential query results in case date_paid meta field is used. + * This way we will only register payments made max 2 month after the order created date. + */ + $max_end = new \WC_DateTime( $args['end'] ); + $max_end->modify( '+2 months' ); + + $joins[] = "LEFT JOIN {$wpdb->postmeta} AS mt3 ON ( {$wpdb->posts}.ID = mt3.post_id AND mt3.meta_key = '_date_paid' )"; + + $where_date_sql = $wpdb->prepare( + "( {$wpdb->posts}.post_date >= %s AND {$wpdb->posts}.post_date <= %s ) AND NOT mt3.post_id IS NULL AND ( + mt3.meta_key = '_date_paid' AND mt3.meta_value >= %s AND mt3.meta_value <= %s + ) OR {$wpdb->posts}.post_parent > 0 AND ( + {$wpdb->posts}.post_date >= %s AND {$wpdb->posts}.post_date <= %s + )", + $args['start'], + $max_end->format( 'Y-m-d' ), + strtotime( $args['start'] ), + $end_adjusted, + $args['start'], + $args['end'] + ); + } + + $join_sql = implode( ' ', $joins ); + + // @codingStandardsIgnoreStart + $sql = $wpdb->prepare( + " + SELECT {$wpdb->posts}.* FROM {$wpdb->posts} + $join_sql + WHERE 1=1 + AND ( {$wpdb->posts}.post_type IN {$post_type_in} ) AND ( {$wpdb->posts}.post_status IN {$post_status_in} ) AND ( {$where_date_sql} ) + AND ( {$where_country_sql} ) + GROUP BY {$wpdb->posts}.ID + ORDER BY {$wpdb->posts}.post_date ASC + LIMIT %d, %d", + $args['offset'], + $args['limit'] + ); + // @codingStandardsIgnoreEnd + + return $sql; + } + + private static function generate_in_query_sql( $values ) { + global $wpdb; + + $in_query = array(); + + foreach ( $values as $value ) { + $in_query[] = $wpdb->prepare( "'%s'", $value ); // phpcs:ignore WordPress.DB.PreparedSQLPlaceholders.QuotedSimplePlaceholder + } + + return '(' . implode( ',', $in_query ) . ')'; + } + + public static function query( $args ) { + global $wpdb; + + $query = self::build_query( $args ); + + Package::extended_log( sprintf( 'Building new query: %s', wc_print_r( $args, true ) ) ); + Package::extended_log( $query ); + + return $wpdb->get_results( $query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } + + public static function cancel( $id ) { + $data = Package::get_report_data( $id ); + $generator = new AsyncReportGenerator( $data['type'], $data ); + $queue = self::get_queue(); + $running = self::get_reports_running(); + + if ( self::is_running( $id ) ) { + $running = array_diff( $running, array( $id ) ); + Package::log( sprintf( 'Cancelled %s', Package::get_report_title( $id ) ) ); + + update_option( 'oss_woocommerce_reports_running', $running, false ); + self::clear_cache(); + $generator->delete(); + } + + /** + * Cancel outstanding events and queue new. + */ + $queue->cancel_all( 'oss_woocommerce_' . $id ); + } + + public static function get_queue() { + return function_exists( 'WC' ) ? WC()->queue() : false; + } + + public static function is_running( $id ) { + $running = self::get_reports_running(); + + if ( in_array( $id, $running, true ) && self::get_queue()->get_next( 'oss_woocommerce_' . $id ) ) { + return true; + } + + return false; + } + + public static function next( $type, $args ) { + /** + * Older versions didn't include refunds as separate orders + */ + if ( ! isset( $args['order_types'] ) ) { + $args['order_types'] = array( 'shop_order' ); + } + + $generator = new AsyncReportGenerator( $type, $args ); + $result = $generator->next(); + $is_empty = false; + $queue = self::get_queue(); + + if ( is_wp_error( $result ) ) { + $is_empty = $result->get_error_message( 'empty' ); + } + + if ( ! $is_empty ) { + $new_args = $generator->get_args(); + + // Increase offset + $new_args['offset'] = (int) $new_args['offset'] + (int) $new_args['limit']; + + $queue->cancel_all( 'oss_woocommerce_' . $generator->get_id() ); + + Package::extended_log( sprintf( 'Starting new queue: %s', wc_print_r( $new_args, true ) ) ); + + $queue->schedule_single( + time() + 10, + 'oss_woocommerce_' . $generator->get_id(), + array( 'args' => $new_args ), + 'oss_woocommerce' + ); + } else { + self::complete( $generator ); + } + } + + /** + * @param AsyncReportGenerator $generator + */ + public static function complete( $generator ) { + $queue = self::get_queue(); + $type = $generator->get_type(); + + /** + * Cancel outstanding events. + */ + $queue->cancel_all( 'oss_woocommerce_' . $generator->get_id() ); + + $report = $generator->complete(); + $status = 'failed'; + + if ( is_a( $report, '\Vendidero\OneStopShop\Report' ) && $report->exists() ) { + $status = 'completed'; + } + + Package::log( sprintf( 'Completed %1$s. Status: %2$s', $report->get_title(), $status ) ); + + self::maybe_stop_report( $report->get_id() ); + + if ( 'observer' === $report->get_type() ) { + self::update_observer( $report ); + } + } + + /** + * @param Report $report + */ + protected static function update_observer( $report ) { + $end = $report->get_date_end(); + $year = $end->date( 'Y' ); + + if ( ! $observer_report = Package::get_observer_report( $year ) ) { + $observer_report = $report; + } else { + $observer_report->set_net_total( $observer_report->get_net_total( false ) + $report->get_net_total( false ) ); + $observer_report->set_tax_total( $observer_report->get_tax_total( false ) + $report->get_tax_total( false ) ); + + foreach ( $report->get_countries() as $country ) { + foreach ( $report->get_tax_rates_by_country( $country ) as $tax_rate ) { + $observer_report->set_country_tax_total( $country, $tax_rate, ( $observer_report->get_country_tax_total( $country, $tax_rate, false ) + $report->get_country_tax_total( $country, $tax_rate, false ) ) ); + $observer_report->set_country_net_total( $country, $tax_rate, ( $observer_report->get_country_net_total( $country, $tax_rate, false ) + $report->get_country_net_total( $country, $tax_rate, false ) ) ); + } + } + + // Delete the old observer report + $observer_report->delete(); + } + + // Delete the tmp report + $report->delete(); + + $observer_report->set_date_requested( $report->get_date_requested() ); + + // Use the last report date as new end date + $observer_report->set_date_end( $report->get_date_end() ); + $observer_report->save(); + + update_option( 'oss_woocommerce_observer_report_' . $year, $observer_report->get_id(), false ); + + do_action( 'oss_woocommerce_updated_observer', $observer_report ); + } + + /** + * @return false|Report + */ + public static function get_running_observer() { + $report = false; + + foreach ( self::get_reports_running() as $id ) { + /** + * Make sure to return the last running observer in case more of one observer exists + * in running queue. + */ + if ( strstr( $id, 'observer_' ) ) { + $report = Package::get_report( $id ); + } + } + + return $report; + } + + public static function maybe_stop_report( $report_id ) { + $reports_running = self::get_reports_running(); + + if ( in_array( $report_id, $reports_running, true ) ) { + $reports_running = array_diff( $reports_running, array( $report_id ) ); + update_option( 'oss_woocommerce_reports_running', $reports_running, false ); + + if ( $queue = self::get_queue() ) { + $queue->cancel_all( 'oss_woocommerce_' . $report_id ); + } + + /** + * Force non-cached running option + */ + wp_cache_delete( 'oss_woocommerce_reports_running', 'options' ); + + return true; + } + + return false; + } + + public static function get_reports_running() { + return (array) get_option( 'oss_woocommerce_reports_running', array() ); + } + + public static function get_timeframe( $type, $date = null, $date_end = null ) { + $date_start = null; + $date_end = is_null( $date_end ) ? null : $date_end; + $start_indicator = is_null( $date ) ? new \WC_DateTime() : $date; + + if ( ! is_a( $start_indicator, 'WC_DateTime' ) && is_numeric( $start_indicator ) ) { + $start_indicator = new \WC_DateTime( '@' . $start_indicator ); + } + + if ( ! is_null( $date_end ) && ! is_a( $date_end, 'WC_DateTime' ) && is_numeric( $date_end ) ) { + $date_end = new \WC_DateTime( '@' . $date_end ); + } + + if ( 'quarterly' === $type ) { + $month = $start_indicator->date( 'n' ); + $quarter = (int) ceil( $month / 3 ); + $start_month = 'Jan'; + $end_month = 'Mar'; + + if ( 2 === $quarter ) { + $start_month = 'Apr'; + $end_month = 'Jun'; + } elseif ( 3 === $quarter ) { + $start_month = 'Jul'; + $end_month = 'Sep'; + } elseif ( 4 === $quarter ) { + $start_month = 'Oct'; + $end_month = 'Dec'; + } + + $date_start = new \WC_DateTime( 'first day of ' . $start_month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + $date_end = new \WC_DateTime( 'last day of ' . $end_month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + } elseif ( 'monthly' === $type ) { + $month = $start_indicator->format( 'M' ); + + $date_start = new \WC_DateTime( 'first day of ' . $month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + $date_end = new \WC_DateTime( 'last day of ' . $month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + } elseif ( 'yearly' === $type ) { + $date_end = clone $start_indicator; + $date_start = clone $start_indicator; + + $date_end->modify( 'last day of dec ' . $start_indicator->format( 'Y' ) . ' midnight' ); + $date_start->modify( 'first day of jan ' . $start_indicator->format( 'Y' ) . ' midnight' ); + } elseif ( 'observer' === $type ) { + $date_start = clone $start_indicator; + $report = Package::get_observer_report( $date_start->format( 'Y' ) ); + + if ( ! $report ) { + // Calculate starting with the first day of the current year until yesterday + $date_end = clone $date_start; + $date_start = new \WC_DateTime( 'first day of jan ' . $start_indicator->format( 'Y' ) . ' midnight' ); + } else { + // In case a report has already been generated lets do only calculate the timeframe between the end of the last report and now + $date_end = clone $date_start; + $date_end->setTime( 0, 0 ); + + $date_start = clone $report->get_date_end(); + $date_start->modify( '+1 day' ); + + if ( $date_start > $date_end ) { + $date_start = clone $date_end; + } + } + } else { + if ( is_null( $date_end ) ) { + $date_end = clone $start_indicator; + $date_end->modify( '-1 year' ); + } + + $date_start = clone $start_indicator; + } + + /** + * Always set start and end time to midnight + */ + if ( $date_start ) { + $date_start->setTime( 0, 0 ); + } + + if ( $date_end ) { + $date_end->setTime( 0, 0 ); + } + + return array( + 'start' => $date_start, + 'end' => $date_end, + ); + } +} diff --git a/packages/one-stop-shop-woocommerce/src/Report.php b/packages/one-stop-shop-woocommerce/src/Report.php new file mode 100644 index 000000000..dcf3519f0 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/Report.php @@ -0,0 +1,330 @@ +set_id( $id ); + + if ( empty( $args ) ) { + $args = (array) get_option( $this->id . '_result', array() ); + } + + $args = wp_parse_args( + $args, + array( + 'countries' => array(), + 'totals' => array(), + 'meta' => array(), + ) + ); + + $args['totals'] = wp_parse_args( + $args['totals'], + array( + 'net_total' => 0, + 'tax_total' => 0, + ) + ); + + $args['meta'] = wp_parse_args( + $args['meta'], + array( + 'date_requested' => null, + 'status' => 'pending', + 'version' => '', + ) + ); + + $this->set_date_requested( $args['meta']['date_requested'] ); + $this->set_status( $args['meta']['status'] ); + $this->set_version( $args['meta']['version'] ); + + $this->args = $args; + } + + public function exists() { + return get_option( $this->id . '_result', false ); + } + + public function get_title() { + $title = Package::get_report_title( $this->get_id() ); + + if ( $this->get_date_requested() ) { + $title = $title . ' @ ' . $this->get_date_requested()->date_i18n(); + } + + return $title; + } + + public function get_url() { + return admin_url( 'admin.php?page=oss-reports&report=' . $this->get_id() ); + } + + public function get_type() { + return $this->type; + } + + public function set_type( $type ) { + $this->set_id_part( $type, 'type' ); + } + + public function set_id( $id ) { + $this->id = $id; + $data = Package::get_report_data( $this->id ); + $this->type = $data['type']; + $this->date_start = $data['date_start']; + $this->date_end = $data['date_end']; + } + + public function set_id_part( $value, $part = 'type' ) { + $data = Package::get_report_data( $this->id ); + $data[ $part ] = $value; + + $this->set_id( Package::get_report_id( $data ) ); + } + + public function get_id() { + return $this->id; + } + + public function get_date_start() { + return $this->date_start; + } + + public function set_date_start( $date ) { + $date = Package::string_to_datetime( $date ); + + $this->set_id_part( $date->format( 'Y-m-d' ), 'date_start' ); + } + + public function get_date_end() { + return $this->date_end; + } + + public function set_date_end( $date ) { + $date = Package::string_to_datetime( $date ); + + $this->set_id_part( $date->format( 'Y-m-d' ), 'date_end' ); + } + + public function get_status() { + return $this->args['meta']['status']; + } + + public function get_version() { + return $this->args['meta']['version']; + } + + public function set_status( $status ) { + $this->args['meta']['status'] = $status; + } + + public function set_version( $version ) { + $this->args['meta']['version'] = $version; + } + + public function get_date_requested() { + return is_null( $this->args['meta']['date_requested'] ) ? null : Package::string_to_datetime( $this->args['meta']['date_requested'] ); + } + + public function set_date_requested( $date ) { + if ( ! empty( $date ) ) { + $date = Package::string_to_datetime( $date ); + } + + $this->args['meta']['date_requested'] = is_a( $date, 'WC_DateTime' ) ? $date->date( 'Y-m-d' ) : null; + } + + public function get_tax_total( $round = true ) { + return $this->maybe_round( $this->args['totals']['tax_total'], $round ); + } + + public function get_net_total( $round = true ) { + return $this->maybe_round( $this->args['totals']['net_total'], $round ); + } + + public function set_tax_total( $total ) { + $this->args['totals']['tax_total'] = wc_format_decimal( floatval( $total ) ); + } + + public function set_net_total( $total ) { + $this->args['totals']['net_total'] = wc_format_decimal( floatval( $total ) ); + } + + public function get_countries() { + return array_keys( $this->args['countries'] ); + } + + public function reset() { + $this->args['countries'] = array(); + + $this->set_net_total( 0 ); + $this->set_tax_total( 0 ); + $this->set_date_requested( new \WC_DateTime() ); + $this->set_status( 'pending' ); + $this->set_version( Package::get_version() ); + + delete_option( $this->id . '_tmp_result' ); + } + + public function get_tax_rates_by_country( $country ) { + $tax_rates = array(); + + if ( array_key_exists( $country, $this->args['countries'] ) ) { + $tax_rates = array_keys( $this->args['countries'][ $country ] ); + } + + return $tax_rates; + } + + public function get_country_tax_total( $country, $tax_rate, $round = true ) { + $tax_total = 0; + + if ( isset( $this->args['countries'][ $country ], $this->args['countries'][ $country ][ "$tax_rate" ] ) ) { + $tax_total = $this->args['countries'][ $country ][ "$tax_rate" ]['tax_total']; + } + + return $this->maybe_round( $tax_total, $round ); + } + + protected function maybe_round( $total, $round = true ) { + $decimals = is_numeric( $round ) ? (int) $round : ''; + + return (float) wc_format_decimal( $total, $round ? $decimals : false ); + } + + public function get_country_net_total( $country, $tax_rate, $round = true ) { + $net_total = 0; + + if ( isset( $this->args['countries'][ $country ], $this->args['countries'][ $country ][ "$tax_rate" ] ) ) { + $net_total = $this->args['countries'][ $country ][ "$tax_rate" ]['net_total']; + } + + return $this->maybe_round( $net_total, $round ); + } + + public function set_country_tax_total( $country, $tax_rate, $tax_total = 0 ) { + if ( ! isset( $this->args['countries'][ $country ] ) ) { + $this->args['countries'][ $country ] = array(); + } + + if ( ! isset( $this->args['countries'][ $country ][ "$tax_rate" ] ) ) { + $this->args['countries'][ $country ][ "$tax_rate" ] = array( + 'net_total' => 0, + 'tax_total' => 0, + ); + } + + $this->args['countries'][ $country ][ "$tax_rate" ]['tax_total'] = $tax_total; + } + + public function set_country_net_total( $country, $tax_rate, $net_total = 0 ) { + if ( ! isset( $this->args['countries'][ $country ] ) ) { + $this->args['countries'][ $country ] = array(); + } + + if ( ! isset( $this->args['countries'][ $country ][ "$tax_rate" ] ) ) { + $this->args['countries'][ $country ][ "$tax_rate" ] = array( + 'net_total' => 0, + 'tax_total' => 0, + ); + } + + $this->args['countries'][ $country ][ "$tax_rate" ]['net_total'] = $net_total; + } + + public function save() { + update_option( $this->id . '_result', $this->args, false ); + + $reports_available = Package::get_report_ids(); + + if ( ! in_array( $this->get_id(), $reports_available[ $this->get_type() ], true ) ) { + // Add new report to start of the list + array_unshift( $reports_available[ $this->get_type() ], $this->get_id() ); + update_option( 'oss_woocommerce_reports', $reports_available, false ); + } + + delete_option( $this->id . '_tmp_result' ); + + Package::clear_caches(); + + return $this->id; + } + + public function delete() { + delete_option( $this->id . '_result' ); + delete_option( $this->id . '_tmp_result' ); + + Queue::maybe_stop_report( $this->get_id() ); + Package::remove_report( $this ); + + if ( 'observer' === $this->get_type() ) { + delete_option( 'oss_woocommerce_observer_report_' . $this->get_date_start()->format( 'Y' ) ); + } + + Package::clear_caches(); + + return true; + } + + public function get_export_link( $export_type = '' ) { + return add_query_arg( + array( + 'action' => 'oss_export_report', + 'export_type' => $export_type, + 'report_id' => $this->get_id(), + ), + wp_nonce_url( admin_url( 'admin-post.php' ), 'oss_export_report' ) + ); + } + + public function get_delete_link() { + return add_query_arg( + array( + 'action' => 'oss_delete_report', + 'report_id' => $this->get_id(), + ), + wp_nonce_url( admin_url( 'admin-post.php' ), 'oss_delete_report' ) + ); + } + + public function get_refresh_link() { + return add_query_arg( + array( + 'action' => 'oss_refresh_report', + 'report_id' => $this->get_id(), + ), + wp_nonce_url( admin_url( 'admin-post.php' ), 'oss_refresh_report' ) + ); + } + + public function get_cancel_link() { + return add_query_arg( + array( + 'action' => 'oss_cancel_report', + 'report_id' => $this->get_id(), + ), + wp_nonce_url( admin_url( 'admin-post.php' ), 'oss_cancel_report' ) + ); + } +} diff --git a/packages/one-stop-shop-woocommerce/src/ReportTable.php b/packages/one-stop-shop-woocommerce/src/ReportTable.php new file mode 100644 index 000000000..7c9206e7c --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/ReportTable.php @@ -0,0 +1,520 @@ + _x( 'Reports', 'oss', 'woocommerce-germanized' ), + 'singular' => _x( 'Report', 'oss', 'woocommerce-germanized' ), + 'screen' => isset( $args['screen'] ) ? $args['screen'] : null, + ) + ); + } + + public function set_default_hidden_columns( $columns, $screen ) { + if ( $this->screen->id === $screen->id ) { + $columns = array_merge( $columns, $this->get_default_hidden_columns() ); + } + + return $columns; + } + + protected function get_default_hidden_columns() { + return array(); + } + + protected function get_hook_prefix() { + return 'oss_woocommerce_admin_reports_table_'; + } + + public function enable_query_removing( $args ) { + $args = array_merge( + $args, + array( + 'changed', + 'bulk_action', + ) + ); + + return $args; + } + + /** + * Handle bulk actions. + * + * @param string $redirect_to URL to redirect to. + * @param string $action Action name. + * @param array $ids List of ids. + * @return string + */ + public function handle_bulk_actions( $action, $ids, $redirect_to ) { + $ids = array_reverse( wc_clean( $ids ) ); + $changed = 0; + + if ( 'delete' === $action ) { + foreach ( $ids as $id ) { + if ( $report = Package::get_report( $id ) ) { + if ( $report->delete() ) { + $changed++; + } + } + } + } + + $changed = apply_filters( "{$this->get_hook_prefix()}bulk_action", $changed, $action, $ids, $redirect_to, $this ); + + if ( $changed ) { + $redirect_to = add_query_arg( + array( + 'changed' => $changed, + 'ids' => join( ',', $ids ), + 'bulk_action' => $action, + ), + $redirect_to + ); + } + + return esc_url_raw( $redirect_to ); + } + + public function output_notices() { + + } + + /** + * Show confirmation message that order status changed for number of orders. + */ + public function set_bulk_notice() { + $number = isset( $_REQUEST['changed'] ) ? absint( $_REQUEST['changed'] ) : 0; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $bulk_action = isset( $_REQUEST['bulk_action'] ) ? wc_clean( wp_unslash( $_REQUEST['bulk_action'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + if ( 'delete' === $bulk_action ) { + $this->add_notice( sprintf( _nx( '%d report deleted.', '%d reports deleted.', $number, 'oss', 'woocommerce-germanized' ), number_format_i18n( $number ) ) ); + } + + do_action( "{$this->get_hook_prefix()}bulk_notice", $bulk_action, $this ); + } + + public function add_notice( $message, $type = 'success' ) { + + } + + /** + * @return bool + */ + public function ajax_user_can() { + return current_user_can( 'manage_woocommerce' ); + } + + public function get_page_option() { + return 'woocommerce_page_oss_reports_per_page'; + } + + public function get_reports( $args ) { + return Package::get_reports( $args ); + } + + /** + * @global array $avail_post_stati + * @global WP_Query $wp_query + * @global int $per_page + * @global string $mode + */ + public function prepare_items() { + global $per_page; + + $per_page = $this->get_items_per_page( $this->get_page_option(), 10 ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited + $per_page = apply_filters( "{$this->get_hook_prefix()}edit_per_page", $per_page ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited + $this->counts = Package::get_report_counts(); + $paged = $this->get_pagenum(); + $report_type = isset( $_REQUEST['type'] ) ? wc_clean( wp_unslash( $_REQUEST['type'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $report_type = in_array( $report_type, array_keys( Package::get_available_report_types( true ) ), true ) ? $report_type : ''; + + $args = array( + 'limit' => $per_page, + 'paginate' => true, + 'offset' => ( $paged - 1 ) * $per_page, + 'count_total' => true, + 'type' => $report_type, + 'include_observer' => 'observer' === $report_type ? true : false, + ); + + $this->items = $this->get_reports( $args ); + + $this->set_pagination_args( + array( + 'total_items' => empty( $args['type'] ) ? array_sum( $this->counts ) : $this->counts[ $args['type'] ], + 'per_page' => $per_page, + ) + ); + } + + /** + */ + public function no_items() { + echo esc_html_x( 'No reports found', 'oss', 'woocommerce-germanized' ); + } + + /** + * Determine if the current view is the "All" view. + * + * @since 4.2.0 + * + * @return bool Whether the current view is the "All" view. + */ + protected function is_base_request() { + $vars = $_GET; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + unset( $vars['paged'] ); + + if ( empty( $vars ) ) { + return true; + } + + return 1 === count( $vars ); + } + + /** + * @global array $locked_post_status This seems to be deprecated. + * @global array $avail_post_stati + * @return array + */ + protected function get_views() { + $type_links = array(); + $num_reports = $this->counts; + $total_reports = array_sum( (array) $num_reports ); + $total_reports = $total_reports - ( isset( $num_reports['observer'] ) ? $num_reports['observer'] : 0 ); + $class = ''; + $all_args = array(); + $include_observers = Package::enable_auto_observer(); + + if ( empty( $class ) && ( $this->is_base_request() || isset( $_REQUEST['all_reports'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $class = 'current'; + } + + $all_inner_html = sprintf( + _nx( + 'All (%s)', + 'All (%s)', + $total_reports, + 'oss', + 'one-stop-shop-woocommerce' + ), + number_format_i18n( $total_reports ) + ); + + $type_links['all'] = $this->get_edit_link( $all_args, $all_inner_html, $class ); + + foreach ( Package::get_available_report_types( $include_observers ) as $type => $title ) { + $class = ''; + + if ( empty( $num_reports[ $type ] ) ) { + continue; + } + + if ( isset( $_REQUEST['type'] ) && $type === $_REQUEST['type'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $class = 'current'; + } + + $type_args = array( + 'type' => $type, + ); + + $type_label = sprintf( + translate_nooped_plural( _nx_noop( $title . ' (%s)', $title . ' (%s)', 'oss', 'woocommerce-germanized' ), $num_reports[ $type ] ), // phpcs:ignore WordPress.WP.I18n.NonSingularStringLiteralPlural,WordPress.WP.I18n.NonSingularStringLiteralSingle + number_format_i18n( $num_reports[ $type ] ) + ); + + $type_links[ $type ] = $this->get_edit_link( $type_args, $type_label, $class ); + } + + return $type_links; + } + + /** + * Helper to create links to edit.php with params. + * + * @since 4.4.0 + * + * @param string[] $args Associative array of URL parameters for the link. + * @param string $label Link text. + * @param string $class Optional. Class attribute. Default empty string. + * @return string The formatted link string. + */ + protected function get_edit_link( $args, $label, $class = '' ) { + $url = add_query_arg( $args, $this->get_main_page() ); + + $class_html = $aria_current = ''; + if ( ! empty( $class ) ) { + $class_html = sprintf( + ' class="%s"', + esc_attr( $class ) + ); + + if ( 'current' === $class ) { + $aria_current = ' aria-current="page"'; + } + } + + return sprintf( + '%s', + esc_url( $url ), + $class_html, + $aria_current, + $label + ); + } + + /** + * @return string + */ + public function current_action() { + if ( isset( $_REQUEST['delete_all'] ) || isset( $_REQUEST['delete_all2'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + return 'delete_all'; + } + + return parent::current_action(); + } + + /** + * @param string $which + */ + protected function extra_tablenav( $which ) { + ?> +
+ render_filters(); + do_action( "{$this->get_hook_prefix()}filters", $which ); + $output = ob_get_clean(); + + if ( ! empty( $output ) ) { + echo $output; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + + submit_button( _x( 'Filter', 'oss', 'woocommerce-germanized' ), '', 'filter_action', false, array( 'id' => 'oss-filter-submit' ) ); + } + } + ?> +
+ '; + $columns['title'] = _x( 'Title', 'oss', 'woocommerce-germanized' ); + $columns['date_start'] = _x( 'Start', 'oss', 'woocommerce-germanized' ); + $columns['date_end'] = _x( 'End', 'oss', 'woocommerce-germanized' ); + $columns['net_total'] = _x( 'Net total', 'oss', 'woocommerce-germanized' ); + $columns['tax_total'] = _x( 'Tax total', 'oss', 'woocommerce-germanized' ); + $columns['status'] = _x( 'Status', 'oss', 'woocommerce-germanized' ); + $columns['actions'] = _x( 'Actions', 'oss', 'woocommerce-germanized' ); + + $columns = apply_filters( "{$this->get_hook_prefix()}columns", $columns ); + + return $columns; + } + + /** + * @return array + */ + protected function get_sortable_columns() { + return array( + 'date_start' => array( 'date_start', false ), + 'date_end' => array( 'date_end', false ), + ); + } + + /** + * Gets the name of the default primary column. + * + * @since 4.3.0 + * + * @return string Name of the default primary column, in this case, 'title'. + */ + protected function get_default_primary_column_name() { + return 'title'; + } + + /** + * Handles the default column output. + * + * @since 4.3.0 + * + * @param Report $report The current shipment object. + * @param string $column_name The current column name. + */ + public function column_default( $report, $column_name ) { + do_action( "{$this->get_hook_prefix()}custom_column", $column_name, $report ); + } + + public function get_main_page() { + return 'admin.php?page=oss-reports'; + } + + /** + * Handles actions. + * + * @since 0.0.1 + * + * @param Report $report The current report object. + */ + protected function column_actions( $report ) { + do_action( "{$this->get_hook_prefix()}actions_start", $report ); + + $actions = Admin::get_report_actions( $report ); + + Admin::render_actions( $actions ); + + do_action( "{$this->get_hook_prefix()}actions_end", $report ); + } + + public function column_cb( $report ) { + ?> + + + get_title(); + + echo '' . esc_html( $title ) . ' '; + } + + /** + * @param Report $report + */ + public function column_status( $report ) { + $status = $report->get_status(); + + return '' . esc_html( Package::get_report_status_title( $status ) ) . ''; + } + + /** + * @param Report $report + */ + public function column_net_total( $report ) { + return wc_price( $report->get_net_total() ); + } + + /** + * @param Report $report + */ + public function column_tax_total( $report ) { + return wc_price( $report->get_tax_total() ); + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Report $report + */ + public function column_date_start( $report ) { + $show_date = $report->get_date_start()->date_i18n( apply_filters( "{$this->get_hook_prefix()}date_format", wc_date_format() ) ); + + printf( + '', + esc_attr( $report->get_date_start()->date( 'c' ) ), + esc_html( $report->get_date_start()->date_i18n( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ) ) ), + esc_html( $show_date ) + ); + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Report $report + */ + public function column_date_end( $report ) { + $show_date = $report->get_date_end()->date_i18n( apply_filters( "{$this->get_hook_prefix()}date_format", wc_date_format() ) ); + + printf( + '', + esc_attr( $report->get_date_end()->date( 'c' ) ), + esc_html( $report->get_date_end()->date_i18n( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ) ) ), + esc_html( $show_date ) + ); + } + + /** + * + * @param Report $report + */ + public function single_row( $report ) { + $GLOBALS['report'] = $report; + $classes = 'report report-' . $report->get_type(); + ?> + + single_row_columns( $report ); ?> + + get_hook_prefix()}bulk_actions", $actions ); + } +} diff --git a/packages/one-stop-shop-woocommerce/src/Settings.php b/packages/one-stop-shop-woocommerce/src/Settings.php new file mode 100644 index 000000000..f983c39e4 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/Settings.php @@ -0,0 +1,203 @@ + _x( 'General', 'oss', 'woocommerce-germanized' ), + ); + } + + public static function get_description() { + return sprintf( _x( 'Find useful options regarding the One Stop Shop procedure here.', 'oss', 'woocommerce-germanized' ) ); + } + + public static function get_help_url() { + return 'https://vendidero.github.io/one-stop-shop-woocommerce/'; + } + + public static function get_settings( $current_section = '' ) { + $settings = array( + array( + 'title' => '', + 'type' => 'title', + 'id' => 'oss_options', + 'desc' => Package::is_integration() ? '' : self::get_description(), + ), + + array( + 'title' => _x( 'OSS status', 'oss', 'woocommerce-germanized' ), + 'desc' => _x( 'Yes, I\'m currently participating in the OSS procedure.', 'oss', 'woocommerce-germanized' ), + 'id' => 'oss_use_oss_procedure', + 'type' => Package::is_integration() ? 'gzd_toggle' : 'checkbox', + 'default' => 'no', + ), + + array( + 'title' => _x( 'Observation', 'oss', 'woocommerce-germanized' ), + 'desc' => _x( 'Automatically observe the delivery threshold of the current year.', 'oss', 'woocommerce-germanized' ) . '

' . _x( 'This option will automatically calculate the amount applicable for the OSS procedure delivery threshold once per day for the current year. The report will only recalculated for the days which are not yet subject to the observation to save processing time.', 'oss', 'woocommerce-germanized' ) . '

', + 'id' => 'oss_enable_auto_observation', + 'type' => Package::is_integration() ? 'gzd_toggle' : 'checkbox', + 'default' => 'yes', + ), + ); + + if ( Package::enable_auto_observer() ) { + $settings = array_merge( + $settings, + array( + array( + 'title' => sprintf( _x( 'Delivery threshold', 'oss', 'woocommerce-germanized' ) ), + 'id' => 'oss_delivery_threshold', + 'type' => 'html', + 'html' => self::get_observer_report_html(), + ), + ) + ); + } + + $settings = array_merge( + $settings, + array( + array( + 'title' => _x( 'Participation', 'oss', 'woocommerce-germanized' ), + 'id' => 'oss_switch', + 'type' => 'html', + 'html' => self::get_oss_switch_html(), + ), + + array( + 'title' => _x( 'Report Order Date', 'oss', 'woocommerce-germanized' ), + 'desc' => '

' . _x( 'Select the relevant order date to be used to determine whether to include an order in a report.', 'oss', 'woocommerce-germanized' ) . '

', + 'id' => 'oss_report_date_type', + 'type' => 'select', + 'default' => 'date_paid', + 'options' => array( + 'date_paid' => _x( 'Date paid', 'oss', 'woocommerce-germanized' ), + 'date_created' => _x( 'Date created', 'oss', 'woocommerce-germanized' ), + ), + ), + ) + ); + + if ( Helper::oss_procedure_is_enabled() && wc_prices_include_tax() ) { + $settings = array_merge( + $settings, + array( + array( + 'title' => _x( 'Fixed gross prices', 'oss', 'woocommerce-germanized' ), + 'desc' => _x( 'Apply the same gross price regardless of the tax rate for EU countries.', 'oss', 'woocommerce-germanized' ) . '

' . _x( 'This option will make sure that your customers pay the same price no matter the tax rate (based on the country chosen) to be applied.', 'oss', 'woocommerce-germanized' ) . '

', + 'id' => 'oss_fixed_gross_prices', + 'type' => Package::is_integration() ? 'gzd_toggle' : 'checkbox', + 'default' => 'yes', + ), + array( + 'title' => _x( 'Third countries', 'oss', 'woocommerce-germanized' ), + 'desc' => _x( 'Apply the same gross price for third countries too.', 'oss', 'woocommerce-germanized' ), + 'id' => 'oss_fixed_gross_prices_for_third_countries', + 'type' => Package::is_integration() ? 'gzd_toggle' : 'checkbox', + 'default' => 'no', + 'custom_attributes' => array( + 'data-show_if_oss_fixed_gross_prices' => '', + ), + ), + ) + ); + } + + $settings = array_merge( + $settings, + array( + array( + 'type' => 'sectionend', + 'id' => 'oss_options', + ), + ) + ); + + return $settings; + } + + public static function get_oss_switch_link() { + return add_query_arg( array( 'action' => 'oss_switch_procedure' ), wp_nonce_url( admin_url( 'admin-post.php' ), 'oss_switch_procedure' ) ); + } + + protected static function get_oss_switch_html() { + ob_start(); + ?> +

+ + + +

+

+ get_url() ) . '">' . esc_html_x( 'See status', 'oss', 'woocommerce-germanized' ) . '' : '' . esc_html_x( 'Start initial report', 'oss', 'woocommerce-germanized' ) . ''; + $status_text = sprintf( ( $running ? esc_html_x( 'Report not yet completed. %s', 'oss', 'woocommerce-germanized' ) : esc_html_x( 'Report not yet started. %s', 'oss', 'woocommerce-germanized' ) ), $status_link ); + ob_start(); + ?> +

+ get_net_total() >= Package::get_delivery_threshold() ) { + $total_class = 'observer-total-red'; + } elseif ( $observer_report->get_net_total() >= Package::get_delivery_notification_threshold() ) { + $total_class = 'observer-total-orange'; + } + + ob_start(); + ?> +

get_net_total() ); ?> get_date_end() ) ); ?>

+

Find out more about the calculation.', 'oss', 'woocommerce-germanized' ), 'https://vendidero.github.io/one-stop-shop-woocommerce/report-calculation' ) ); ?>

+ id = 'oss'; + $this->label = _x( 'OSS', 'oss', 'woocommerce-germanized' ); + + parent::__construct(); + } + + public function output() { + echo '

' . esc_html_x( 'One Stop Shop', 'oss', 'woocommerce-germanized' ) . ' ' . esc_html_x( 'Reports', 'oss', 'woocommerce-germanized' ) . ' ' . esc_html_x( 'Learn More', 'oss', 'woocommerce-germanized' ) . '

'; + + parent::output(); + } + + /** + * Get sections. + * + * @return array + */ + public function get_sections() { + $sections = Settings::get_sections(); + + return apply_filters( 'woocommerce_get_sections_' . $this->id, $sections ); + } + + public function save() { + Settings::before_save(); + parent::save(); + Settings::after_save(); + } + + /** + * Get settings array. + * + * @return array + */ + public function get_settings( $current_section = '' ) { + $settings = Settings::get_settings( $current_section ); + + return apply_filters( 'woocommerce_get_settings_' . $this->id, $settings ); + } + + public function get_settings_for_section_core( $section_id ) { + return Settings::get_settings( $section_id ); + } +} diff --git a/packages/one-stop-shop-woocommerce/src/Tax.php b/packages/one-stop-shop-woocommerce/src/Tax.php new file mode 100644 index 000000000..80148fd43 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/src/Tax.php @@ -0,0 +1,561 @@ +needs_shipping() ) { + return true; + } + + return false; + } + + protected static function filter_cart_items_calculated_totals( $item ) { + return isset( $item['line_total'] ); + } + + /** + * As prices may change based on the customers address and VAT status (e.g. exempt) + * it is necessary to make sure that shipping tax is recalculated too in case shipping costs include taxes. + */ + public static function invalidate_shipping_session( $cart ) { + if ( apply_filters( 'oss_shipping_costs_include_taxes', false ) ) { + if ( $cart ) { + $items = array_values( array_filter( $cart->get_cart(), array( __CLASS__, 'filter_cart_items_available_for_shipping' ) ) ); + $items_calculated = array_values( array_filter( $items, array( __CLASS__, 'filter_cart_items_calculated_totals' ) ) ); + + /** + * Make sure totals have already been calculated (for all items) to prevent missing array key warnings + * while calling WC_Cart::get_shipping_packages() + */ + if ( count( $items ) > 0 && $items == $items_calculated ) { // phpcs:ignore WordPress.PHP.StrictComparisons.LooseComparison + foreach ( $cart->get_shipping_packages() as $package_key => $package ) { + $session_key = "shipping_for_package_{$package_key}"; + + unset( WC()->session->$session_key ); + } + } + } + } + } + + /** + * In case the order/customer is a VAT exempt, use the base address as tax location. + * + * @param $location + * + * @return array|mixed + */ + public static function vat_exempt_taxable_address( $location ) { + if ( Helper::current_request_has_vat_exempt() ) { + $location = array( + WC()->countries->get_base_country(), + WC()->countries->get_base_state(), + WC()->countries->get_base_postcode(), + WC()->countries->get_base_city(), + ); + } + + return $location; + } + + /** + * Maybe reset order item tax class while recalculating order totals as the product tax class + * filter will only work for the initial add to order event. + * + * @param \WC_Order_Item $item + * + * @return void + */ + public static function maybe_filter_order_item_tax_class( $item ) { + if ( is_a( $item, 'WC_Order_Item_Product' ) && ( $order = $item->get_order() ) ) { + if ( isset( $_POST['action'] ) && 'woocommerce_calc_line_taxes' === wc_clean( wp_unslash( $_POST['action'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + if ( $product = $item->get_product() ) { + $taxable_address = Helper::get_order_taxable_location( $order ); + + if ( isset( $taxable_address[0] ) && ! empty( $taxable_address[0] ) ) { + $address = array( + 'country' => $taxable_address[0], + 'state' => isset( $taxable_address[1] ) ? $taxable_address[1] : '', + 'postcode' => isset( $taxable_address[2] ) ? $taxable_address[2] : '', + 'city' => isset( $taxable_address[3] ) ? $taxable_address[3] : '', + ); + + $tax_class = self::get_product_tax_class_by_country( $product, $address ); + + if ( $tax_class !== $item->get_tax_class() && apply_filters( 'oss_woocommerce_switch_order_item_tax_class', true, $tax_class, $item ) ) { + $item->set_tax_class( $tax_class ); + } + } + } + } + } + } + + /** + * @param $tax_class + * @param \WC_Product $product + */ + public static function filter_tax_class( $tax_class, $product ) { + $taxable_address = Helper::get_taxable_location(); + + if ( isset( $taxable_address[0] ) && ! empty( $taxable_address[0] ) && WC()->countries->get_base_country() !== $taxable_address[0] ) { + $address = array( + 'country' => $taxable_address[0], + 'state' => isset( $taxable_address[1] ) ? $taxable_address[1] : '', + 'postcode' => isset( $taxable_address[2] ) ? $taxable_address[2] : '', + 'city' => isset( $taxable_address[3] ) ? $taxable_address[3] : '', + ); + + $tax_class = self::get_product_tax_class_by_country( $product, $address, $tax_class ); + } + + return $tax_class; + } + + /** + * @param \WC_Product_Variation $variation + * @param $i + */ + public static function save_variation_options( $variation, $i ) { + $parent = wc_get_product( $variation->get_parent_id() ); + $tax_classes = self::get_product_tax_classes( $variation, $parent, 'edit' ); + $parent_tax_classes = self::get_product_tax_classes( $parent ); + $product_tax_class = $variation->get_tax_class(); + + $posted = isset( $_POST['variable_tax_class_by_countries'][ $i ] ) ? wc_clean( (array) wp_unslash( $_POST['variable_tax_class_by_countries'][ $i ] ) ) : array(); // phpcs:ignore WordPress.Security.NonceVerification.Missing + $new_classes = isset( $_POST['variable_tax_class_by_countries_new_tax_class'][ $i ] ) ? wc_clean( (array) wp_unslash( $_POST['variable_tax_class_by_countries_new_tax_class'][ $i ] ) ) : array(); // phpcs:ignore WordPress.Security.NonceVerification.Missing + $new_countries = isset( $_POST['variable_tax_class_by_countries_new_countries'][ $i ] ) ? wc_clean( (array) wp_unslash( $_POST['variable_tax_class_by_countries_new_countries'][ $i ] ) ) : array(); // phpcs:ignore WordPress.Security.NonceVerification.Missing + + foreach ( $tax_classes as $country => $tax_class ) { + // Maybe delete missing tax classes (e.g. removed by the user) + if ( ! isset( $posted[ $country ] ) || 'parent' === $posted[ $country ] ) { + unset( $tax_classes[ $country ] ); + } else { + $tax_classes[ $country ] = $posted[ $country ]; + } + } + + foreach ( $new_countries as $key => $country ) { + if ( empty( $country ) ) { + continue; + } + + if ( ! array_key_exists( $country, $tax_classes ) && isset( $new_classes[ $key ] ) && 'parent' !== $new_classes[ $key ] ) { + $tax_classes[ $country ] = $new_classes[ $key ]; + } + } + + /** + * Remove tax classes which match the products main tax class or the base country + */ + foreach ( $tax_classes as $country => $tax_class ) { + if ( $tax_class === $product_tax_class || WC()->countries->get_base_country() === $country ) { + unset( $tax_classes[ $country ] ); + } elseif ( isset( $parent_tax_classes[ $country ] ) && $parent_tax_classes[ $country ] === $tax_class ) { + unset( $tax_classes[ $country ] ); + } elseif ( 'parent' === $tax_class ) { + unset( $tax_classes[ $country ] ); + } + } + + if ( empty( $tax_classes ) ) { + $variation->delete_meta_data( '_tax_class_by_countries' ); + } else { + $variation->update_meta_data( '_tax_class_by_countries', $tax_classes ); + } + } + + /** + * @param \WC_Product $product + */ + public static function save_product_options( $product ) { + $tax_classes = self::get_product_tax_classes( $product ); + $product_tax_class = $product->get_tax_class(); + + $posted = isset( $_POST['_tax_class_by_countries'] ) ? wc_clean( (array) wp_unslash( $_POST['_tax_class_by_countries'] ) ) : array(); // phpcs:ignore WordPress.Security.NonceVerification.Missing + $new_classes = isset( $_POST['_tax_class_by_countries_new_tax_class'] ) ? wc_clean( (array) wp_unslash( $_POST['_tax_class_by_countries_new_tax_class'] ) ) : array(); // phpcs:ignore WordPress.Security.NonceVerification.Missing + $new_countries = isset( $_POST['_tax_class_by_countries_new_countries'] ) ? wc_clean( (array) wp_unslash( $_POST['_tax_class_by_countries_new_countries'] ) ) : array(); // phpcs:ignore WordPress.Security.NonceVerification.Missing + + foreach ( $tax_classes as $country => $tax_class ) { + // Maybe delete missing tax classes (e.g. removed by the user) + if ( ! isset( $posted[ $country ] ) ) { + unset( $tax_classes[ $country ] ); + } else { + $tax_classes[ $country ] = $posted[ $country ]; + } + } + + foreach ( $new_countries as $key => $country ) { + if ( empty( $country ) ) { + continue; + } + + if ( ! array_key_exists( $country, $tax_classes ) && isset( $new_classes[ $key ] ) ) { + $tax_classes[ $country ] = $new_classes[ $key ]; + } + } + + /** + * Remove tax classes which match the products main tax class or the base country + */ + foreach ( $tax_classes as $country => $tax_class ) { + if ( $tax_class === $product_tax_class || WC()->countries->get_base_country() === $country ) { + unset( $tax_classes[ $country ] ); + } + } + + if ( empty( $tax_classes ) ) { + $product->delete_meta_data( '_tax_class_by_countries' ); + } else { + $product->update_meta_data( '_tax_class_by_countries', $tax_classes ); + } + } + + /** + * @param $loop + * @param $variation_data + * @param \WP_Post $variation + */ + public static function variation_tax_product_options( $loop, $variation_data, $variation ) { + global $product_object; + + if ( ! $variation = wc_get_product( $variation ) ) { + return; + } + + $tax_classes = self::get_product_tax_classes( $variation, $product_object, 'edit' ); + $countries_left = self::get_selectable_countries(); + + if ( ! empty( $tax_classes ) ) { + foreach ( $tax_classes as $country => $tax_class ) { + $countries_left = array_diff_key( $countries_left, array( $country => '' ) ); + + woocommerce_wp_select( + array( + 'id' => "variable_tax_class_by_countries{$loop}_{$country}", + 'name' => "variable_tax_class_by_countries[{$loop}][{$country}]", + 'value' => $tax_class, + 'label' => sprintf( _x( 'Tax class (%s)', 'oss', 'woocommerce-germanized' ), $country ), + 'options' => array( 'parent' => _x( 'Same as parent', 'oss', 'woocommerce-germanized' ) ) + wc_get_product_tax_class_options(), + 'wrapper_class' => 'oss-tax-class-by-country-field form-row form-row-full', + 'description' => '' . _x( 'remove', 'oss', 'woocommerce-germanized' ) . '', + ) + ); + } + } + ?> +
+ +

+ + + +

+ +
+

+ + + + + +

+
+ _x( 'EU-wide', 'oss', 'woocommerce-germanized' ) ); + + return $eu + $countries; + } + + protected static function get_country_name( $country_code ) { + $country_name = $country_code; + $countries = WC()->countries ? WC()->countries->get_countries() : array(); + + if ( 'EU-wide' === $country_code ) { + $country_name = _x( 'EU-wide', 'oss', 'woocommerce-germanized' ); + } elseif ( isset( $countries[ $country_code ] ) ) { + $country_name = $countries[ $country_code ]; + } + + return $country_name; + } + + public static function tax_product_options() { + global $product_object; + + $tax_classes = self::get_product_tax_classes( $product_object ); + $countries_left = self::get_selectable_countries(); + + if ( ! empty( $tax_classes ) ) { + foreach ( $tax_classes as $country => $tax_class ) { + $countries_left = array_diff_key( $countries_left, array( $country => '' ) ); + + woocommerce_wp_select( + array( + 'id' => '_tax_class_by_countries_' . $country, + 'name' => '_tax_class_by_countries[' . $country . ']', + 'value' => $tax_class, + 'label' => sprintf( _x( 'Tax class (%s)', 'oss', 'woocommerce-germanized' ), $country ), + 'options' => wc_get_product_tax_class_options(), + 'description' => '' . _x( 'remove', 'oss', 'woocommerce-germanized' ) . '', + ) + ); + } + } + + ?> +
+ +

+ + + +

+ +
+

+ + + + + +

+
+ '', + 'state' => '', + 'postcode' => '', + 'city' => '', + ) + ); + + $tax_class = false !== $default ? $default : $product->get_tax_class(); + $postcode = wc_normalize_postcode( $address['postcode'] ); + $filter_tax_class = true; + + /** + * Prevent tax class adjustment for GB (except Norther Ireland via postcode detection) + */ + if ( 'GB' === $address['country'] && ( empty( $postcode ) || 'BT' !== substr( $postcode, 0, 2 ) ) ) { + $filter_tax_class = false; + } + + if ( apply_filters( 'oss_woocommerce_switch_product_tax_class', $filter_tax_class, $product, $address['country'], $postcode, $default ) ) { + $cache_suffix = '_oss_tax_class_' . md5( sprintf( '%s+%s+%s+%s+%s', $address['country'], $address['state'], $address['city'], $postcode, $product->get_id() ) ); + $cache_key = \WC_Cache_Helper::get_cache_prefix( 'product_' . $product->get_id() ) . $cache_suffix; + $cache_key_tax = \WC_Cache_Helper::get_cache_prefix( 'taxes' ) . $cache_suffix; + $matched_tax_cache = wp_cache_get( $cache_key_tax, 'taxes' ); + $matched_tax_class = false !== $matched_tax_cache ? wp_cache_get( $cache_key, 'products' ) : false; + + if ( false === $matched_tax_class ) { + $tax_classes = self::get_product_tax_classes( $product ); + $tax_class_slugs = Helper::get_tax_class_slugs(); + + if ( array_key_exists( $address['country'], $tax_classes ) ) { + $tax_class = $tax_classes[ $address['country'] ]; + } elseif ( isset( $tax_classes['EU-wide'] ) ) { + $tax_class = $tax_classes['EU-wide']; + } + + if ( $tax_class_slugs['super-reduced'] === $tax_class ) { + $tax_rates = \WC_Tax::find_rates( + array( + 'country' => $address['country'], + 'state' => $address['state'], + 'city' => $address['city'], + 'postcode' => $postcode, + 'tax_class' => $tax_class, + ) + ); + + /** + * Country does not seem to support this tax class - fallback to the reduced tax class + */ + if ( empty( $tax_rates ) ) { + $tax_class = $tax_class_slugs['reduced']; + } + } + + if ( $tax_class_slugs['greater-reduced'] === $tax_class ) { + $tax_rates = \WC_Tax::find_rates( + array( + 'country' => $address['country'], + 'state' => $address['state'], + 'city' => $address['city'], + 'postcode' => $postcode, + 'tax_class' => $tax_class, + ) + ); + + /** + * Country does not seem to support this tax class - fallback to the reduced tax class + */ + if ( empty( $tax_rates ) ) { + $tax_class = $tax_class_slugs['reduced']; + } + } + + if ( $tax_class_slugs['reduced'] === $tax_class ) { + $tax_rates = \WC_Tax::find_rates( + array( + 'country' => $address['country'], + 'state' => $address['state'], + 'city' => $address['city'], + 'postcode' => $postcode, + 'tax_class' => $tax_class, + ) + ); + + /** + * Country does not seem to support this tax class - fallback to the standard tax class + */ + if ( empty( $tax_rates ) ) { + $tax_class = $tax_class_slugs['standard']; + } + } + + /** + * This cache entry depends on both the tax and product data. + */ + wp_cache_set( $cache_key_tax, $cache_key, 'taxes' ); + wp_cache_set( $cache_key, $tax_class, 'products' ); + } else { + $tax_class = $matched_tax_class; + } + } + + return $tax_class; + } + + /** + * @param \WC_Product $product + */ + public static function get_product_tax_classes( $product, $parent = false, $context = 'view' ) { + $tax_classes = $product->get_meta( '_tax_class_by_countries', true ); + $tax_classes = ( ! is_array( $tax_classes ) || empty( $tax_classes ) ) ? array() : $tax_classes; + + /** + * Merge with parent tax classes + */ + if ( is_a( $product, 'WC_Product_Variation' ) ) { + $parent = $parent ? $parent : wc_get_product( $product->get_parent_id() ); + + if ( $parent ) { + $parent_tax_classes = self::get_product_tax_classes( $parent ); + $tax_classes = array_replace_recursive( $parent_tax_classes, $tax_classes ); + + foreach ( $tax_classes as $country => $tax_class ) { + $parent_tax_class = isset( $parent_tax_classes[ $country ] ) ? $parent_tax_classes[ $country ] : false; + + if ( 'view' === $context && 'parent' === $tax_class ) { + if ( $parent_tax_class ) { + $tax_classes[ $country ] = $parent_tax_class; + } else { + unset( $tax_classes[ $country ] ); + } + } elseif ( 'edit' === $context && $tax_class === $parent_tax_class ) { + $tax_classes[ $country ] = 'parent'; + } + } + } + } + + return $tax_classes; + } +} diff --git a/packages/one-stop-shop-woocommerce/templates/emails/admin-delivery-threshold.php b/packages/one-stop-shop-woocommerce/templates/emails/admin-delivery-threshold.php new file mode 100644 index 000000000..4661feea4 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/templates/emails/admin-delivery-threshold.php @@ -0,0 +1,42 @@ + + + +

OSS Settings Panel for details.', 'oss', 'woocommerce-germanized' ), wc_price( \Vendidero\OneStopShop\Package::get_delivery_notification_threshold() ), esc_url( \Vendidero\OneStopShop\Settings::get_settings_url() ) ) ); ?>

+ +

+ +
    +
  • : get_date_start()->format( wc_date_format() ) ); ?> - get_date_end()->format( wc_date_format() ) ); ?>
  • +
  • : get_net_total() ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>
  • +
  • : get_tax_total() ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>
  • +
+ + +get_url() ); + +/** + * Show user-defined additional content - this is set in each email's settings. + */ +if ( $additional_content ) { + echo esc_html( wp_strip_all_tags( wptexturize( $additional_content ) ) ); + echo "\n\n----------------------------------------\n\n"; +} + +echo "\n----------------------------------------\n\n"; + +echo wp_kses_post( apply_filters( 'woocommerce_email_footer_text', get_option( 'woocommerce_email_footer_text' ) ) ); diff --git a/packages/one-stop-shop-woocommerce/wpml-config.xml b/packages/one-stop-shop-woocommerce/wpml-config.xml new file mode 100644 index 000000000..6de03ace2 --- /dev/null +++ b/packages/one-stop-shop-woocommerce/wpml-config.xml @@ -0,0 +1,5 @@ + + + _tax_class_by_countries + + \ No newline at end of file diff --git a/packages/woocommerce-eu-tax-helper/src/Helper.php b/packages/woocommerce-eu-tax-helper/src/Helper.php new file mode 100644 index 000000000..420dd2fdd --- /dev/null +++ b/packages/woocommerce-eu-tax-helper/src/Helper.php @@ -0,0 +1,911 @@ +countries->get_european_union_countries(); + + return $countries; + } + + public static function get_eu_vat_countries() { + return apply_filters( 'woocommerce_eu_tax_helper_eu_vat_countries', WC()->countries->get_european_union_countries( 'eu_vat' ) ); + } + + public static function is_northern_ireland( $country, $postcode = '' ) { + if ( 'GB' === $country && 'BT' === strtoupper( substr( trim( $postcode ), 0, 2 ) ) ) { + return true; + } elseif ( 'IX' === $country ) { + return true; + } + + return false; + } + + public static function is_eu_vat_country( $country, $postcode = '' ) { + $country = wc_strtoupper( $country ); + $postcode = wc_normalize_postcode( $postcode ); + $is_eu_vat_country = in_array( $country, self::get_eu_vat_countries(), true ); + + if ( self::is_northern_ireland( $country, $postcode ) ) { + $is_eu_vat_country = true; + } elseif ( self::is_eu_vat_postcode_exemption( $country, $postcode ) ) { + $is_eu_vat_country = false; + } + + return apply_filters( 'woocommerce_eu_tax_helper_is_eu_vat_country', $is_eu_vat_country, $country, $postcode ); + } + + public static function is_third_country( $country, $postcode = '' ) { + $is_third_country = true; + + /** + * In case the base country is within EU consider all non-EU VAT countries as third countries. + * In any other case consider every non-base-country as third country. + */ + if ( in_array( self::get_base_country(), self::get_eu_vat_countries(), true ) ) { + $is_third_country = ! self::is_eu_vat_country( $country, $postcode ); + } else { + $is_third_country = self::get_base_country() !== $country; + } + + return apply_filters( 'woocommerce_eu_tax_helper_is_third_country', $is_third_country, $country, $postcode ); + } + + public static function is_eu_country( $country ) { + return in_array( $country, self::get_eu_countries(), true ); + } + + public static function is_eu_vat_postcode_exemption( $country, $postcode = '' ) { + $country = wc_strtoupper( $country ); + $postcode = wc_normalize_postcode( $postcode ); + $exemptions = self::get_vat_postcode_exemptions_by_country(); + $is_exempt = false; + + if ( ! empty( $postcode ) && in_array( $country, self::get_eu_vat_countries(), true ) ) { + if ( array_key_exists( $country, $exemptions ) ) { + $wildcards = wc_get_wildcard_postcodes( $postcode, $country ); + + foreach ( $exemptions[ $country ] as $exempt_postcode ) { + if ( in_array( $exempt_postcode, $wildcards, true ) ) { + $is_exempt = true; + break; + } + } + } + } + + return $is_exempt; + } + + /** + * Get VAT exemptions (of EU countries) for certain postcodes (e.g. canary islands) + * + * @see https://www.hk24.de/produktmarken/beratung-service/recht-und-steuern/steuerrecht/umsatzsteuer-mehrwertsteuer/umsatzsteuer-mehrwertsteuer-international/verfahrensrecht/territoriale-besonderheiten-umsatzsteuer-zollrecht-1167674 + * @see https://github.com/woocommerce/woocommerce/issues/5143 + * @see https://ec.europa.eu/taxation_customs/business/vat/eu-vat-rules-topic/territorial-status-eu-countries-certain-territories_en + * + * @return \string[][] + */ + public static function get_vat_postcode_exemptions_by_country( $country = '' ) { + $country = wc_strtoupper( $country ); + + $exemptions = array( + 'DE' => array( + '27498', // Helgoland + '78266', // Büsingen am Hochrhein + ), + 'ES' => array( + '35*', // Canary Islands + '38*', // Canary Islands + '51*', // Ceuta + '52*', // Melilla + ), + 'GR' => array( + '63086', // Mount Athos + '63087', // Mount Athos + ), + 'FR' => array( + '971*', // Guadeloupe + '972*', // Martinique + '973*', // French Guiana + '974*', // Réunion + '976*', // Mayotte + ), + 'IT' => array( + '22060', // Livigno, Campione d’Italia + '23030', // Lake Lugano + ), + 'FI' => array( + '22*', // Aland islands + ), + ); + + if ( empty( $country ) ) { + return $exemptions; + } elseif ( array_key_exists( $country, $exemptions ) ) { + return $exemptions[ $country ]; + } else { + return array(); + } + } + + /** + * @param integer|\WC_Order $order + * + * @return array + */ + public static function get_order_taxable_location( $order ) { + $order = is_a( $order, 'WC_Order' ) ? $order : wc_get_order( $order ); + + $taxable_address = array( + WC()->countries->get_base_country(), + WC()->countries->get_base_state(), + WC()->countries->get_base_postcode(), + WC()->countries->get_base_city(), + ); + + if ( ! $order ) { + return $taxable_address; + } + + $tax_based_on = get_option( 'woocommerce_tax_based_on' ); + + if ( is_a( $order, 'WC_Order_Refund' ) ) { + $order = wc_get_order( $order->get_parent_id() ); + + if ( ! $order ) { + return $taxable_address; + } + } + + /** + * Shipping address data does not exist + */ + if ( 'shipping' === $tax_based_on && ! $order->get_shipping_country() ) { + $tax_based_on = 'billing'; + } + + $is_vat_exempt = apply_filters( 'woocommerce_order_is_vat_exempt', 'yes' === $order->get_meta( 'is_vat_exempt' ), $order ); + + /** + * In case the order is a VAT exempt, calculate net prices based on taxes from base country. + */ + if ( $is_vat_exempt ) { + $tax_based_on = 'base'; + } + + $country = 'shipping' === $tax_based_on ? $order->get_shipping_country() : $order->get_billing_country(); + + if ( 'base' !== $tax_based_on && ! empty( $country ) ) { + $taxable_address = array( + $country, + 'billing' === $tax_based_on ? $order->get_billing_state() : $order->get_shipping_state(), + 'billing' === $tax_based_on ? $order->get_billing_postcode() : $order->get_shipping_postcode(), + 'billing' === $tax_based_on ? $order->get_billing_city() : $order->get_shipping_city(), + ); + } + + return $taxable_address; + } + + public static function get_taxable_location() { + $is_admin_order_request = self::is_admin_order_request(); + + if ( $is_admin_order_request ) { + $taxable_address = array( + WC()->countries->get_base_country(), + WC()->countries->get_base_state(), + WC()->countries->get_base_postcode(), + WC()->countries->get_base_city(), + ); + + if ( isset( $_POST['order_id'] ) && ( $order = wc_get_order( absint( $_POST['order_id'] ) ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $taxable_address = self::get_order_taxable_location( $order ); + } + + return $taxable_address; + } else { + return \WC_Tax::get_tax_location(); + } + } + + public static function is_admin_order_ajax_request() { + $order_actions = array( 'woocommerce_calc_line_taxes', 'woocommerce_save_order_items', 'add_coupon_discount', 'refund_line_items', 'delete_refund' ); + + return isset( $_POST['action'], $_POST['order_id'] ) && ( strstr( wc_clean( wp_unslash( $_POST['action'] ) ), '_order_' ) || in_array( wc_clean( wp_unslash( $_POST['action'] ) ), $order_actions, true ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + public static function is_admin_order_request() { + return is_admin() && current_user_can( 'edit_shop_orders' ) && self::is_admin_order_ajax_request(); + } + + public static function current_request_has_vat_exempt() { + $is_admin_order_request = self::is_admin_order_request(); + $is_vat_exempt = false; + + if ( $is_admin_order_request ) { + if ( $order = wc_get_order( absint( $_POST['order_id'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing,WordPress.Security.ValidatedSanitizedInput.InputNotValidated + $is_vat_exempt = apply_filters( 'woocommerce_order_is_vat_exempt', 'yes' === $order->get_meta( 'is_vat_exempt' ), $order ); + } + } else { + if ( WC()->customer && WC()->customer->is_vat_exempt() ) { + $is_vat_exempt = true; + } + } + + return $is_vat_exempt; + } + + public static function get_base_country() { + if ( WC()->countries ) { + return WC()->countries->get_base_country(); + } else { + return wc_get_base_location()['country']; + } + } + + /** + * Returns a list of EU countries except base country. + * + * @return string[] + */ + public static function get_non_base_eu_countries( $include_gb = false ) { + $countries = WC()->countries->get_european_union_countries( 'eu_vat' ); + + /** + * Include GB to allow Northern Ireland + */ + if ( $include_gb && ! in_array( 'GB', $countries, true ) ) { + $countries = array_merge( $countries, array( 'GB' ) ); + } + + $base_country = self::get_base_country(); + $countries = array_diff( $countries, array( $base_country ) ); + + return $countries; + } + + public static function country_supports_eu_vat( $country, $postcode = '' ) { + return self::is_eu_vat_country( $country, $postcode ); + } + + public static function import_oss_tax_rates( $tax_class_slug_names = array() ) { + self::import_tax_rates_internal( true, $tax_class_slug_names ); + } + + public static function import_default_tax_rates( $tax_class_slug_names = array() ) { + self::import_tax_rates_internal( false, $tax_class_slug_names ); + } + + public static function import_tax_rates( $tax_class_slug_names = array() ) { + self::import_tax_rates_internal( self::oss_procedure_is_enabled(), $tax_class_slug_names ); + } + + protected static function parse_tax_class_slug_names( $tax_class_slug_names = array() ) { + return wp_parse_args( + $tax_class_slug_names, + array( + 'reduced' => apply_filters( 'woocommerce_eu_tax_helper_tax_class_reduced_name', __( 'Reduced rate', 'woocommerce' ) ), // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + 'greater-reduced' => apply_filters( 'woocommerce_eu_tax_helper_tax_class_greater_reduced_name', _x( 'Greater reduced rate', 'tax-helper-tax-class-name', 'woocommerce-germanized' ) ), + 'super-reduced' => apply_filters( 'woocommerce_eu_tax_helper_tax_class_super_reduced_name', _x( 'Super reduced rate', 'tax-helper-tax-class-name', 'woocommerce-germanized' ) ), + ) + ); + } + + protected static function import_tax_rates_internal( $is_oss = true, $tax_class_slug_names = array() ) { + self::clear_cache(); + + $tax_class_slugs = self::get_tax_class_slugs( $tax_class_slug_names ); + $tax_class_slug_names = self::parse_tax_class_slug_names( $tax_class_slug_names ); + $eu_rates = self::get_eu_tax_rates(); + + foreach ( $tax_class_slugs as $tax_class_type => $class ) { + /** + * Maybe create missing tax classes + */ + if ( false === $class ) { + switch ( $tax_class_type ) { + case 'reduced': + /* translators: Do not translate */ + \WC_Tax::create_tax_class( $tax_class_slug_names['reduced'] ); // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + break; + case 'greater-reduced': + \WC_Tax::create_tax_class( $tax_class_slug_names['greater-reduced'] ); + break; + case 'super-reduced': + \WC_Tax::create_tax_class( $tax_class_slug_names['super-reduced'] ); + break; + } + } + + $new_rates = array(); + + foreach ( $eu_rates as $country => $rates_data ) { + + /** + * Use base country rates in case OSS is disabled + */ + if ( ! $is_oss ) { + $base_country = self::get_base_country(); + + if ( isset( $eu_rates[ $base_country ] ) ) { + /** + * In case the country includes multiple rules (e.g. postcode exempts) by default + * do only use the last rule (which does not include exempts) to construct non-base country tax rules. + */ + if ( $base_country !== $country ) { + $base_country_base_rate = array_values( array_slice( $eu_rates[ $base_country ], -1 ) )[0]; + + foreach ( $rates_data as $key => $rate_data ) { + $rates_data[ $key ] = array_replace_recursive( $rate_data, $base_country_base_rate ); + + foreach ( $tax_class_slugs as $tmp_class_type => $class_data ) { + /** + * Do not include tax classes which are not supported by the base country. + */ + if ( isset( $rates_data[ $key ][ $tmp_class_type ] ) && ! isset( $base_country_base_rate[ $tmp_class_type ] ) ) { + unset( $rates_data[ $key ][ $tmp_class_type ] ); + } elseif ( isset( $rates_data[ $key ][ $tmp_class_type ] ) ) { + /** + * Replace tax class data with base data to make sure that reduced + * classes have the same dimensions + */ + $rates_data[ $key ][ $tmp_class_type ] = $base_country_base_rate[ $tmp_class_type ]; + + /** + * In case this is an exempt make sure to replace with zero tax rates + */ + if ( isset( $rate_data['is_exempt'] ) && $rate_data['is_exempt'] ) { + if ( is_array( $rates_data[ $key ][ $tmp_class_type ] ) ) { + foreach ( $rates_data[ $key ][ $tmp_class_type ] as $k => $rate ) { + $rates_data[ $key ][ $tmp_class_type ][ $k ] = 0; + } + } else { + $rates_data[ $key ][ $tmp_class_type ] = 0; + } + } + } + } + } + } + } else { + continue; + } + } + + /** + * Each country may contain multiple tax rates + */ + foreach ( $rates_data as $rates ) { + + $rates = wp_parse_args( + $rates, + array( + 'name' => '', + 'postcodes' => array(), + 'reduced' => array(), + ) + ); + + if ( ! empty( $rates['postcode'] ) ) { + foreach ( $rates['postcode'] as $postcode ) { + $tax_rate = self::get_single_tax_rate_data( $tax_class_type, $rates, $country, $postcode ); + + if ( false !== $tax_rate ) { + $new_rates[] = $tax_rate; + } + } + } else { + $tax_rate = self::get_single_tax_rate_data( $tax_class_type, $rates, $country ); + + if ( false !== $tax_rate ) { + $new_rates[] = $tax_rate; + } + } + } + } + + self::import_rates( $new_rates, $class ); + } + } + + private static function get_single_tax_rate_data( $tax_class_type, $rates, $country, $postcode = '' ) { + $rates = wp_parse_args( + $rates, + array( + 'name' => '', + 'reduced' => array(), + ) + ); + + $single_rate = array( + 'name' => $rates['name'], + 'rate' => false, + 'country' => $country, + 'postcode' => $postcode, + ); + + switch ( $tax_class_type ) { + case 'greater-reduced': + if ( count( $rates['reduced'] ) > 1 ) { + $single_rate['rate'] = $rates['reduced'][1]; + } + break; + case 'reduced': + if ( ! empty( $rates['reduced'] ) ) { + $single_rate['rate'] = $rates['reduced'][0]; + } + break; + default: + if ( isset( $rates[ $tax_class_type ] ) ) { + $single_rate['rate'] = $rates[ $tax_class_type ]; + } + break; + } + + if ( false === $single_rate['rate'] ) { + return false; + } + + return $single_rate; + } + + protected static function clear_cache() { + $cache_key = \WC_Cache_Helper::get_cache_prefix( 'taxes' ) . 'eu_tax_helper_tax_class_slugs'; + + wp_cache_delete( $cache_key, 'taxes' ); + } + + public static function get_tax_class_slugs( $tax_class_slug_names = array() ) { + $tax_class_slug_names = self::parse_tax_class_slug_names( $tax_class_slug_names ); + $cache_key = \WC_Cache_Helper::get_cache_prefix( 'taxes' ) . 'eu_tax_helper_tax_class_slugs'; + $slugs = wp_cache_get( $cache_key, 'taxes' ); + + if ( false === $slugs ) { + $reduced_tax_class = false; + $greater_reduced_tax_class = false; + $super_reduced_tax_class = false; + $tax_classes = \WC_Tax::get_tax_class_slugs(); + + /** + * Try to determine the reduced tax rate class + */ + foreach ( $tax_classes as $slug ) { + if ( strstr( $slug, 'virtual' ) ) { + continue; + } + + if ( ! $greater_reduced_tax_class && strstr( $slug, sanitize_title( 'Greater reduced rate' ) ) ) { + $greater_reduced_tax_class = $slug; + } elseif ( ! $greater_reduced_tax_class && strstr( $slug, sanitize_title( $tax_class_slug_names['greater-reduced'] ) ) ) { + $greater_reduced_tax_class = $slug; + } elseif ( ! $super_reduced_tax_class && strstr( $slug, sanitize_title( 'Super reduced rate' ) ) ) { + $super_reduced_tax_class = $slug; + } elseif ( ! $super_reduced_tax_class && strstr( $slug, sanitize_title( $tax_class_slug_names['super-reduced'] ) ) ) { + $super_reduced_tax_class = $slug; + } elseif ( ! $reduced_tax_class && strstr( $slug, sanitize_title( 'Reduced rate' ) ) ) { + $reduced_tax_class = $slug; + } elseif ( ! $reduced_tax_class && strstr( $slug, sanitize_title( $tax_class_slug_names['reduced'] ) ) ) { // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + $reduced_tax_class = $slug; + } elseif ( ! $reduced_tax_class && strstr( $slug, 'reduced' ) && ! $reduced_tax_class ) { + $reduced_tax_class = $slug; + } + } + + $slugs = array( + 'reduced' => $reduced_tax_class, + 'greater-reduced' => $greater_reduced_tax_class, + 'super-reduced' => $super_reduced_tax_class, + 'standard' => '', + ); + + wp_cache_set( $cache_key, $slugs, 'taxes' ); + } + + return apply_filters( 'woocommerce_eu_tax_helper_tax_rate_class_slugs', $slugs ); + } + + public static function get_tax_type_by_country_rate( $rate_percentage, $country ) { + $country = strtoupper( $country ); + + /** + * Map northern ireland to GB + */ + if ( 'XI' === $country ) { + $country = 'GB'; + } + + $eu_rates = self::get_eu_tax_rates(); + $tax_type = 'standard'; + + if ( array_key_exists( $country, $eu_rates ) ) { + $rates = $eu_rates[ $country ]; + + foreach ( $rates as $rate ) { + foreach ( $rate as $tax_rate_type => $tax_rate_percent ) { + if ( ( is_array( $tax_rate_percent ) && in_array( $rate_percentage, $tax_rate_percent, true ) ) || (float) $tax_rate_percent === (float) $rate_percentage ) { + $tax_type = $tax_rate_type; + break; + } + } + } + } + + return apply_filters( 'woocommerce_eu_tax_helper_country_rate_tax_type', $tax_type, $country, $rate_percentage ); + } + + public static function get_eu_tax_rates() { + /** + * @see https://europa.eu/youreurope/business/taxation/vat/vat-rules-rates/index_en.htm + * + * Include Great Britain to allow including Norther Ireland + */ + $rates = array( + 'AT' => array( + array( + 'standard' => 20, + 'reduced' => array( 10, 13 ), + ), + ), + 'BE' => array( + array( + 'standard' => 21, + 'reduced' => array( 6, 12 ), + ), + ), + 'BG' => array( + array( + 'standard' => 20, + 'reduced' => array( 9 ), + ), + ), + 'CY' => array( + array( + 'standard' => 19, + 'reduced' => array( 5, 9 ), + ), + ), + 'CZ' => array( + array( + 'standard' => 21, + 'reduced' => array( 10, 15 ), + ), + ), + 'DE' => array( + array( + 'standard' => 19, + 'reduced' => array( 7 ), + ), + ), + 'DK' => array( + array( + 'standard' => 25, + 'reduced' => array(), + ), + ), + 'EE' => array( + array( + 'standard' => 20, + 'reduced' => array( 9 ), + ), + ), + 'GR' => array( + array( + 'standard' => 24, + 'reduced' => array( 6, 13 ), + ), + ), + 'ES' => array( + array( + 'standard' => 21, + 'reduced' => array( 10 ), + 'super-reduced' => 4, + ), + ), + 'FI' => array( + array( + 'standard' => 24, + 'reduced' => array( 10, 14 ), + ), + ), + 'FR' => array( + array( + 'standard' => 20, + 'reduced' => array( 5.5, 10 ), + 'super-reduced' => 2.1, + ), + ), + 'HR' => array( + array( + 'standard' => 25, + 'reduced' => array( 5, 13 ), + ), + ), + 'HU' => array( + array( + 'standard' => 27, + 'reduced' => array( 5, 18 ), + ), + ), + 'IE' => array( + array( + 'standard' => 23, + 'reduced' => array( 9, 13.5 ), + 'super-reduced' => 4.8, + ), + ), + 'IT' => array( + array( + 'standard' => 22, + 'reduced' => array( 5, 10 ), + 'super-reduced' => 4, + ), + ), + 'LT' => array( + array( + 'standard' => 21, + 'reduced' => array( 5, 9 ), + ), + ), + 'LU' => array( + array( + 'standard' => 16, + 'reduced' => array( 7 ), + 'super-reduced' => 3, + ), + ), + 'LV' => array( + array( + 'standard' => 21, + 'reduced' => array( 12, 5 ), + ), + ), + 'MC' => array( + array( + 'standard' => 20, + 'reduced' => array( 5.5, 10 ), + 'super-reduced' => 2.1, + ), + ), + 'MT' => array( + array( + 'standard' => 18, + 'reduced' => array( 5, 7 ), + ), + ), + 'NL' => array( + array( + 'standard' => 21, + 'reduced' => array( 9 ), + ), + ), + 'PL' => array( + array( + 'standard' => 23, + 'reduced' => array( 5, 8 ), + ), + ), + 'PT' => array( + array( + // Madeira + 'postcode' => array( '90*', '91*', '92*', '93*', '94*' ), + 'standard' => 22, + 'reduced' => array( 5, 12 ), + 'name' => _x( 'Madeira', 'tax-helper', 'woocommerce-germanized' ), + ), + array( + // Acores + 'postcode' => array( '95*', '96*', '97*', '98*', '99*' ), + 'standard' => 18, + 'reduced' => array( 4, 9 ), + 'name' => _x( 'Acores', 'tax-helper', 'woocommerce-germanized' ), + ), + array( + 'standard' => 23, + 'reduced' => array( 6, 13 ), + ), + ), + 'RO' => array( + array( + 'standard' => 19, + 'reduced' => array( 5, 9 ), + ), + ), + 'SE' => array( + array( + 'standard' => 25, + 'reduced' => array( 6, 12 ), + ), + ), + 'SI' => array( + array( + 'standard' => 22, + 'reduced' => array( 9.5 ), + ), + ), + 'SK' => array( + array( + 'standard' => 20, + 'reduced' => array( 10 ), + ), + ), + 'GB' => array( + array( + 'standard' => 20, + 'reduced' => array( 5 ), + 'postcode' => array( 'BT*' ), + 'name' => _x( 'Northern Ireland', 'tax-helper', 'woocommerce-germanized' ), + ), + ), + ); + + foreach ( self::get_vat_postcode_exemptions_by_country() as $country => $exempt_postcodes ) { + if ( array_key_exists( $country, $rates ) ) { + $default_rate = array_values( $rates[ $country ] )[0]; + + $postcode_exempt = array( + 'postcode' => $exempt_postcodes, + 'standard' => 0, + 'reduced' => count( $default_rate['reduced'] ) > 1 ? array( 0, 0 ) : array( 0 ), + 'name' => _x( 'Exempt', 'tax-helper-rate-import', 'woocommerce-germanized' ), + 'is_exempt' => true, + ); + + if ( array_key_exists( 'super-reduced', $default_rate ) ) { + $postcode_exempt['super-reduced'] = 0; + } + + // Prepend before other tax rates + $rates[ $country ] = array_merge( array( $postcode_exempt ), $rates[ $country ] ); + } + } + + return $rates; + } + + /** + * @param \stdClass $rate + * + * @return bool + */ + public static function tax_rate_is_northern_ireland( $rate ) { + if ( 'GB' === $rate->tax_rate_country && isset( $rate->postcode ) && ! empty( $rate->postcode ) ) { + foreach ( $rate->postcode as $postcode ) { + if ( self::is_northern_ireland( $rate->tax_rate_country, $postcode ) ) { + return true; + } + } + } + + return false; + } + + public static function import_rates( $rates, $tax_class = '' ) { + global $wpdb; + + $eu_countries = self::get_eu_vat_countries(); + + /** + * Delete EU tax rates and make sure tax rate locations are deleted too + */ + foreach ( \WC_Tax::get_rates_for_tax_class( $tax_class ) as $rate_id => $rate ) { + if ( in_array( $rate->tax_rate_country, $eu_countries, true ) || self::tax_rate_is_northern_ireland( $rate ) || ( 'GB' === $rate->tax_rate_country && 'GB' !== self::get_base_country() ) ) { + \WC_Tax::_delete_tax_rate( $rate_id ); + } + } + + $count = 0; + + foreach ( $rates as $rate ) { + $rate = wp_parse_args( + $rate, + array( + 'rate' => 0, + 'country' => '', + 'postcode' => '', + 'name' => '', + ) + ); + + $iso = wc_strtoupper( $rate['country'] ); + $vat_desc = $iso; + + if ( ! empty( $rate['name'] ) ) { + $vat_desc = $vat_desc . ' ' . $rate['name']; + } + + $vat_rate = wc_format_decimal( $rate['rate'], false, true ); + + $tax_rate_name = apply_filters( 'woocommerce_eu_tax_helper_import_tax_rate_name', sprintf( _x( 'VAT %1$s %% %2$s', 'tax-helper-rate-import', 'woocommerce-germanized' ), $vat_rate, $vat_desc ), $rate['rate'], $iso, $tax_class, $rate ); + + $_tax_rate = array( + 'tax_rate_country' => $iso, + 'tax_rate_state' => '', + 'tax_rate' => (string) number_format( (float) wc_clean( $rate['rate'] ), 4, '.', '' ), + 'tax_rate_name' => $tax_rate_name, + 'tax_rate_compound' => 0, + 'tax_rate_priority' => 1, + 'tax_rate_order' => $count++, + 'tax_rate_shipping' => ( strstr( $tax_class, 'virtual' ) ? 0 : 1 ), + 'tax_rate_class' => $tax_class, + ); + + $new_tax_rate_id = \WC_Tax::_insert_tax_rate( $_tax_rate ); + + if ( ! empty( $rate['postcode'] ) ) { + \WC_Tax::_update_tax_rate_postcodes( $new_tax_rate_id, $rate['postcode'] ); + } + } + } + + /** + * @param $rate_id + * @param \WC_Order $order + */ + public static function get_tax_rate_percent( $rate_id, $order ) { + $taxes = $order->get_taxes(); + $percentage = null; + + foreach ( $taxes as $tax ) { + if ( (int) $tax->get_rate_id() === (int) $rate_id ) { + if ( is_callable( array( $tax, 'get_rate_percent' ) ) ) { + $percentage = $tax->get_rate_percent(); + } + } + } + + /** + * WC_Order_Item_Tax::get_rate_percent returns null by default. + * Fallback to global tax rates (DB) in case the percentage is not available within order data. + */ + if ( is_null( $percentage ) || '' === $percentage ) { + $rate_percentage = self::get_tax_rate_percentage( $rate_id ); + + if ( false !== $rate_percentage ) { + $percentage = $rate_percentage; + } + } + + if ( ! is_numeric( $percentage ) ) { + $percentage = 0; + } + + return $percentage; + } + + public static function get_tax_rate_percentage( $rate_id ) { + $percentage = false; + + if ( is_callable( array( 'WC_Tax', 'get_rate_percent_value' ) ) ) { + $percentage = \WC_Tax::get_rate_percent_value( $rate_id ); + } elseif ( is_callable( array( 'WC_Tax', 'get_rate_percent' ) ) ) { + $percentage = filter_var( \WC_Tax::get_rate_percent( $rate_id ), FILTER_SANITIZE_NUMBER_FLOAT, FILTER_FLAG_ALLOW_FRACTION ); + } + + return $percentage; + } +} diff --git a/packages/woocommerce-germanized-dhl/assets/css/admin.css b/packages/woocommerce-germanized-dhl/assets/css/admin.css new file mode 100644 index 000000000..30fa46b20 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/admin.css @@ -0,0 +1,48 @@ +.germanized-create-label .wc-gzd-shipment-im-additional-services p.label { + margin-top: 10px; + width: 100%; + display: block; + margin-bottom: 5px; + font-weight: bold; } + +.germanized-create-label .wc-gzd-dhl-im-product-data { + margin-top: 2em; + min-width: 700px; + margin-left: -1rem !important; + margin-right: -1rem !important; } + .germanized-create-label .wc-gzd-dhl-im-product-data .column { + padding-left: 1rem !important; + padding-right: 1rem !important; } + .germanized-create-label .wc-gzd-dhl-im-product-data .column p:first-child { + margin-top: 1.5em !important; } + .germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-price { + background: #ffd633; + border-radius: 4px; + padding: .5em 1em; } + .germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-price .amount { + font-size: 18px; + font-weight: bold; } + .germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-price .price-suffix { + display: block; + font-size: 11px; + line-height: 15px; } + .germanized-create-label .wc-gzd-dhl-im-product-data .col-dimensions { + color: #999; } + .germanized-create-label .wc-gzd-dhl-im-product-data .col-preview .image-preview img { + height: auto; + max-height: 140px; } + .germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-information-text, .germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-description { + font-size: 11px; + color: #999; + line-height: 1.5em; } + +.germanized-create-label .show-services-trigger { + font-weight: bold; + margin-top: 15px; + margin-bottom: 0; + display: block; + text-align: right; + vertical-align: middle; + line-height: 20px; } + .germanized-create-label .show-services-trigger a { + text-decoration: none; } \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/admin.min.css b/packages/woocommerce-germanized-dhl/assets/css/admin.min.css new file mode 100644 index 000000000..59195f1cd --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/admin.min.css @@ -0,0 +1 @@ +.germanized-create-label .wc-gzd-shipment-im-additional-services p.label{margin-top:10px;width:100%;display:block;margin-bottom:5px;font-weight:700}.germanized-create-label .wc-gzd-dhl-im-product-data{margin-top:2em;min-width:700px;margin-left:-1rem!important;margin-right:-1rem!important}.germanized-create-label .wc-gzd-dhl-im-product-data .column{padding-left:1rem!important;padding-right:1rem!important}.germanized-create-label .wc-gzd-dhl-im-product-data .column p:first-child{margin-top:1.5em!important}.germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-price{background:#ffd633;border-radius:4px;padding:.5em 1em}.germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-price .amount{font-size:18px;font-weight:700}.germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-price .price-suffix{display:block;font-size:11px;line-height:15px}.germanized-create-label .wc-gzd-dhl-im-product-data .col-dimensions{color:#999}.germanized-create-label .wc-gzd-dhl-im-product-data .col-preview .image-preview img{height:auto;max-height:140px}.germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-description,.germanized-create-label .wc-gzd-dhl-im-product-data .wc-gzd-dhl-im-product-information-text{font-size:11px;color:#999;line-height:1.5em}.germanized-create-label .show-services-trigger{font-weight:700;margin-top:15px;margin-bottom:0;display:block;text-align:right;vertical-align:middle;line-height:20px}.germanized-create-label .show-services-trigger a{text-decoration:none} \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/admin.scss b/packages/woocommerce-germanized-dhl/assets/css/admin.scss new file mode 100644 index 000000000..32abfae9b --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/admin.scss @@ -0,0 +1,76 @@ +.germanized-create-label { + .wc-gzd-shipment-im-additional-services { + p.label { + margin-top: 10px; + width: 100%; + display: block; + margin-bottom: 5px; + font-weight: bold; + } + } + + .wc-gzd-dhl-im-product-data { + margin-top: 2em; + min-width: 700px; + margin-left: -1rem !important; + margin-right: -1rem !important; + + .column { + padding-left: 1rem !important; + padding-right: 1rem !important; + + p:first-child { + margin-top: 1.5em !important; + } + } + + .wc-gzd-dhl-im-product-price { + background: #ffd633; + border-radius: 4px; + padding: .5em 1em; + + .amount { + font-size: 18px; + font-weight: bold; + } + .price-suffix { + display: block; + font-size: 11px; + line-height: 15px; + } + } + + .col-dimensions { + color: #999; + } + + .col-preview { + .image-preview { + img { + height: auto; + max-height: 140px; + } + } + } + + .wc-gzd-dhl-im-product-information-text, .wc-gzd-dhl-im-product-description { + font-size: 11px; + color: #999; + line-height: 1.5em; + } + } + + .show-services-trigger { + font-weight: bold; + margin-top: 15px; + margin-bottom: 0; + display: block; + text-align: right; + vertical-align: middle; + line-height: 20px; + + a { + text-decoration: none; + } + } +} \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.css b/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.css new file mode 100644 index 000000000..1fad22ff7 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.css @@ -0,0 +1,155 @@ +#dhl-parcel-finder-wrapper { + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + transform: translateZ(0); + width: 100%; + height: 100%; + left: 0; + position: fixed; + top: 0; + z-index: 99992; + visibility: hidden; } + #dhl-parcel-finder-wrapper * { + box-sizing: border-box; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-bg-overlay { + background: #1e1e1e; + opacity: 0; + transition-duration: inherit; + transition-property: opacity; + bottom: 0; + left: 0; + position: absolute; + right: 0; + top: 0; } + #dhl-parcel-finder-wrapper.open { + visibility: visible; } + #dhl-parcel-finder-wrapper.open #dhl-parcel-finder-bg-overlay { + opacity: .87; + transition-timing-function: cubic-bezier(0.22, 0.61, 0.36, 1); } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-inner { + bottom: 0; + left: 0; + position: absolute; + right: 0; + top: 0; + overflow: hidden; + z-index: 99994; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-inner-wrapper { + padding: 6px 6px 0; + display: block; + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + height: 100%; + left: 0; + outline: none; + overflow: auto; + -webkit-overflow-scrolling: touch; + position: absolute; + text-align: center; + top: 0; + transition-property: transform,opacity; + white-space: normal; + width: 100%; + z-index: 99994; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-inner-wrapper::before { + content: ""; + display: inline-block; + height: 100%; + margin-right: -.25em; + vertical-align: middle; + width: 0; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-inner-wrapper #dhl-parcel-finder { + width: 95%; + height: 95%; + display: inline-block; + background: #fff; + margin: 0 0 6px; + max-width: 100%; + overflow: auto; + padding: 34px; + position: relative; + text-align: left; + vertical-align: middle; } + #dhl-parcel-finder-wrapper .dhl-parcel-finder-close { + background: transparent; + border: 0; + border-radius: 0; + color: #555; + cursor: pointer; + height: 44px; + margin: 0; + padding: 6px; + position: absolute; + right: 0; + top: 0; + width: 44px; + z-index: 10; } + #dhl-parcel-finder-wrapper .dhl-parcel-finder-close svg { + fill: transparent; + opacity: .8; + stroke: currentColor; + stroke-width: 1.5; + transition: stroke .1s; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-map { + width: 100%; + height: 85%; + position: relative !important; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content #bodyContent { + line-height: 1.5em; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content address { + margin-bottom: 0; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content .parcel-title { + padding-top: 0; + margin-bottom: .5em; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content .parcel-subtitle { + font-size: 0.8125rem; + color: #767676; + font-weight: 800; + letter-spacing: 0.15em; + text-transform: uppercase; + padding: 1em 0 0; } + #dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content .dhl-parcelshop-select-btn { + width: 100%; + margin-top: 10px; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form { + display: flex; + flex-wrap: wrap; + justify-content: flex-start; + align-items: center; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field { + margin-right: 1.5em; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.large { + min-width: 20%; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.finder-pickup-type { + margin-right: 25px; + min-width: 150px; + display: flex; + align-items: center; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.finder-pickup-type.hidden { + display: none; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.finder-pickup-type .icon { + width: 40px; + height: 40px; + display: inline-block; + background-repeat: no-repeat; + background-size: contain; + margin-left: 10px; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button { + text-align: right; + margin-right: 0; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button .button { + width: 100%; + margin: 0; } + +@media (max-width: 700px) { + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field { + width: 100%; + margin-right: 0; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.packstation, #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.parcelshop { + width: auto; + margin-right: 1.5em; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button { + text-align: left; } + #dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button .button { + width: 100%; + margin: 0; } } \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.min.css b/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.min.css new file mode 100644 index 000000000..555775f6e --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.min.css @@ -0,0 +1 @@ +#dhl-parcel-finder-wrapper{-webkit-backface-visibility:hidden;backface-visibility:hidden;transform:translateZ(0);width:100%;height:100%;left:0;position:fixed;top:0;z-index:99992;visibility:hidden}#dhl-parcel-finder-wrapper *{box-sizing:border-box}#dhl-parcel-finder-wrapper #dhl-parcel-finder-bg-overlay{background:#1e1e1e;opacity:0;transition-duration:inherit;transition-property:opacity;bottom:0;left:0;position:absolute;right:0;top:0}#dhl-parcel-finder-wrapper.open{visibility:visible}#dhl-parcel-finder-wrapper.open #dhl-parcel-finder-bg-overlay{opacity:.87;transition-timing-function:cubic-bezier(.22,.61,.36,1)}#dhl-parcel-finder-wrapper #dhl-parcel-finder-inner{bottom:0;left:0;position:absolute;right:0;top:0;overflow:hidden;z-index:99994}#dhl-parcel-finder-wrapper #dhl-parcel-finder-inner-wrapper{padding:6px 6px 0;display:block;-webkit-backface-visibility:hidden;backface-visibility:hidden;height:100%;left:0;outline:0;overflow:auto;-webkit-overflow-scrolling:touch;position:absolute;text-align:center;top:0;transition-property:transform,opacity;white-space:normal;width:100%;z-index:99994}#dhl-parcel-finder-wrapper #dhl-parcel-finder-inner-wrapper::before{content:"";display:inline-block;height:100%;margin-right:-.25em;vertical-align:middle;width:0}#dhl-parcel-finder-wrapper #dhl-parcel-finder-inner-wrapper #dhl-parcel-finder{width:95%;height:95%;display:inline-block;background:#fff;margin:0 0 6px;max-width:100%;overflow:auto;padding:34px;position:relative;text-align:left;vertical-align:middle}#dhl-parcel-finder-wrapper .dhl-parcel-finder-close{background:0 0;border:0;border-radius:0;color:#555;cursor:pointer;height:44px;margin:0;padding:6px;position:absolute;right:0;top:0;width:44px;z-index:10}#dhl-parcel-finder-wrapper .dhl-parcel-finder-close svg{fill:transparent;opacity:.8;stroke:currentColor;stroke-width:1.5;transition:stroke .1s}#dhl-parcel-finder-wrapper #dhl-parcel-finder-map{width:100%;height:85%;position:relative!important}#dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content #bodyContent{line-height:1.5em}#dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content address{margin-bottom:0}#dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content .parcel-title{padding-top:0;margin-bottom:.5em}#dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content .parcel-subtitle{font-size:.8125rem;color:#767676;font-weight:800;letter-spacing:.15em;text-transform:uppercase;padding:1em 0 0}#dhl-parcel-finder-wrapper #dhl-parcel-finder-map #parcel-content .dhl-parcelshop-select-btn{width:100%;margin-top:10px}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form{display:flex;flex-wrap:wrap;justify-content:flex-start;align-items:center}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field{margin-right:1.5em}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.large{min-width:20%}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.finder-pickup-type{margin-right:25px;min-width:150px;display:flex;align-items:center}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.finder-pickup-type.hidden{display:none}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.finder-pickup-type .icon{width:40px;height:40px;display:inline-block;background-repeat:no-repeat;background-size:contain;margin-left:10px}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button{text-align:right;margin-right:0}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button .button{width:100%;margin:0}@media (max-width:700px){#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field{width:100%;margin-right:0}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.packstation,#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field.parcelshop{width:auto;margin-right:1.5em}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button{text-align:left}#dhl-parcel-finder-wrapper form#dhl-parcel-finder-form .form-field#dhl-search-button .button{width:100%;margin:0}} \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.scss b/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.scss new file mode 100644 index 000000000..9a7c9833b --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/parcel-finder.scss @@ -0,0 +1,218 @@ +#dhl-parcel-finder-wrapper { + backface-visibility: hidden; + transform: translateZ(0); + width: 100%; + height: 100%; + left: 0; + position: fixed; + top: 0; + z-index: 99992; + visibility: hidden; + + * { + box-sizing: border-box; + } + + #dhl-parcel-finder-bg-overlay { + background: #1e1e1e; + opacity: 0; + transition-duration: inherit; + transition-property: opacity; + bottom: 0; + left: 0; + position: absolute; + right: 0; + top: 0; + } + + &.open { + visibility: visible; + + #dhl-parcel-finder-bg-overlay { + opacity: .87; + transition-timing-function: cubic-bezier(.22,.61,.36,1); + } + } + + #dhl-parcel-finder-inner { + bottom: 0; + left: 0; + position: absolute; + right: 0; + top: 0; + overflow: hidden; + z-index: 99994; + } + + #dhl-parcel-finder-inner-wrapper { + padding: 6px 6px 0; + display: block; + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + height: 100%; + left: 0; + outline: none; + overflow: auto; + -webkit-overflow-scrolling: touch; + position: absolute; + text-align: center; + top: 0; + transition-property: transform,opacity; + white-space: normal; + width: 100%; + z-index: 99994; + + &::before { + content: ""; + display: inline-block; + height: 100%; + margin-right: -.25em; + vertical-align: middle; + width: 0; + } + + #dhl-parcel-finder { + width: 95%; + height: 95%; + display: inline-block; + background: #fff; + margin: 0 0 6px; + max-width: 100%; + overflow: auto; + padding: 34px; + position: relative; + text-align: left; + vertical-align: middle; + } + } + + .dhl-parcel-finder-close { + background: transparent; + border: 0; + border-radius: 0; + color: #555; + cursor: pointer; + height: 44px; + margin: 0; + padding: 6px; + position: absolute; + right: 0; + top: 0; + width: 44px; + z-index: 10; + + svg { + fill: transparent; + opacity: .8; + stroke: currentColor; + stroke-width: 1.5; + transition: stroke .1s; + } + } + + #dhl-parcel-finder-map { + width: 100%; + height: 85%; + position: relative !important; + + #parcel-content { + + #bodyContent { + line-height: 1.5em; + } + + address { + margin-bottom: 0; + } + + .parcel-title { + padding-top: 0; + margin-bottom: .5em; + } + + .parcel-subtitle { + font-size: 0.8125rem; + color: #767676; + font-weight: 800; + letter-spacing: 0.15em; + text-transform: uppercase; + padding: 1em 0 0; + } + + .dhl-parcelshop-select-btn { + width: 100%; + margin-top: 10px; + } + } + } + + form#dhl-parcel-finder-form { + display: flex; + flex-wrap: wrap; + justify-content: flex-start; + align-items: center; + + .form-field { + margin-right: 1.5em; + + &.large { + min-width: 20%; + } + + &.finder-pickup-type { + margin-right: 25px; + min-width: 150px; + display: flex; + align-items: center; + + &.hidden { + display: none; + } + + .icon { + width: 40px; + height: 40px; + display: inline-block; + background-repeat: no-repeat; + background-size: contain; + margin-left: 10px; + } + } + + &#dhl-search-button { + text-align: right; + margin-right: 0; + + .button { + width: 100%; + margin: 0; + } + } + } + } +} + +@media( max-width: 700px ) { + #dhl-parcel-finder-wrapper { + form#dhl-parcel-finder-form { + .form-field { + width: 100%; + margin-right: 0; + + &.packstation, &.parcelshop { + width: auto; + margin-right: 1.5em; + } + + &#dhl-search-button { + text-align: left; + + .button { + width: 100%; + margin: 0; + } + } + } + } + } +} \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/preferred-services.css b/packages/woocommerce-germanized-dhl/assets/css/preferred-services.css new file mode 100644 index 000000000..1e9a62142 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/preferred-services.css @@ -0,0 +1,156 @@ +#tiptip_holder { + display: none; + z-index: 8675309; + position: absolute; + top: 0; + /*rtl:ignore*/ + left: 0; } + #tiptip_holder.tip_top { + padding-bottom: 5px; } + #tiptip_holder.tip_top #tiptip_arrow_inner { + margin-top: -7px; + margin-left: -6px; + border-top-color: #333; } + #tiptip_holder.tip_bottom { + padding-top: 5px; } + #tiptip_holder.tip_bottom #tiptip_arrow_inner { + margin-top: -5px; + margin-left: -6px; + border-bottom-color: #333; } + #tiptip_holder.tip_right { + padding-left: 5px; } + #tiptip_holder.tip_right #tiptip_arrow_inner { + margin-top: -6px; + margin-left: -5px; + border-right-color: #333; } + #tiptip_holder.tip_left { + padding-right: 5px; } + #tiptip_holder.tip_left #tiptip_arrow_inner { + margin-top: -6px; + margin-left: -7px; + border-left-color: #333; } + +#tiptip_content { + color: #fff; + font-size: 0.8em; + max-width: 150px; + background: #333; + text-align: center; + border-radius: 3px; + padding: 0.618em 1em; + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2); } + #tiptip_content code { + padding: 1px; + background: #888; } + +#tiptip_arrow, +#tiptip_arrow_inner { + position: absolute; + border-color: transparent; + border-style: solid; + border-width: 6px; + height: 0; + width: 0; } + +.dhl-preferred-service-content { + margin-top: 1em; } + .dhl-preferred-service-content .dhl-hidden { + display: none; } + .dhl-preferred-service-content .dhl-preferred-service-cost { + font-size: .9em; } + .dhl-preferred-service-content .dhl-preferred-service-item { + margin-bottom: 1em; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-logo { + margin-bottom: 1em; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-logo img { + margin: 0; + padding: 0; + max-height: 100px; + max-width: 100px; + background: #FFCC00; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-title { + font-weight: bold; + font-size: 1em; + margin-bottom: .5em; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-cost { + margin-bottom: .5em; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-desc { + font-size: .9em; + margin-bottom: .5em; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times { + display: flex; + flex-direction: row; + flex-wrap: wrap; + margin: 0; + padding: 0; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li { + flex-basis: 10%; + display: inline-block; + text-align: center; + padding: 10px 0 0; + margin: 0 8px 8px 0; + background-color: #e3e3e3; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li label { + position: relative; + display: flex; + flex-direction: column; + flex-wrap: wrap; + padding: 5px 10px; + font-size: .9em; + font-weight: bold; + background-color: #eef4f2; + cursor: pointer; + margin: 0; + color: #5f7285; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li label .dhl-preferred-time-title { + font-size: 1.2em; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li input[type=radio] { + opacity: 0; + width: 1px; + height: 1px; + position: absolute; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li input[type=radio]:checked ~ label { + background-color: #FFCC00; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times.dhl-preferred-service-time li { + flex-grow: inherit; + flex-basis: inherit; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times.dhl-preferred-service-time li label .dhl-preferred-time-title { + font-size: 1em; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-data input[type=text] { + width: 100%; + margin-bottom: .5em; } + .dhl-preferred-service-content .dhl-preferred-service-item .woocommerce-help-tip { + background: #ffcc00; + display: inline-block; + font-size: 1em; + font-style: normal; + height: 18px; + padding: 3px; + margin-top: -5px; + margin-left: 5px; + border-radius: 50%; + line-height: 18px; + position: relative; + vertical-align: middle; + width: 18px; } + .dhl-preferred-service-content .dhl-preferred-service-item .woocommerce-help-tip::after { + position: absolute; + top: 0; + left: 0; + width: 100%; + height: 100%; + text-align: center; + cursor: help; + content: "?"; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-location-types, .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-delivery-types { + list-style: none; + margin: 0; + margin-bottom: 1em; + display: flex; + flex-wrap: wrap; + justify-content: flex-start; + align-items: center; } + .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-location-types li, .dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-delivery-types li { + margin-right: 1em; } + .dhl-preferred-service-content .dhl-preferred-service-item.dhl-preferred-service-header .dhl-preferred-service-title { + font-size: 1.1em; } \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/preferred-services.min.css b/packages/woocommerce-germanized-dhl/assets/css/preferred-services.min.css new file mode 100644 index 000000000..ef224e57d --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/preferred-services.min.css @@ -0,0 +1 @@ +#tiptip_holder{display:none;z-index:8675309;position:absolute;top:0;left:0}#tiptip_holder.tip_top{padding-bottom:5px}#tiptip_holder.tip_top #tiptip_arrow_inner{margin-top:-7px;margin-left:-6px;border-top-color:#333}#tiptip_holder.tip_bottom{padding-top:5px}#tiptip_holder.tip_bottom #tiptip_arrow_inner{margin-top:-5px;margin-left:-6px;border-bottom-color:#333}#tiptip_holder.tip_right{padding-left:5px}#tiptip_holder.tip_right #tiptip_arrow_inner{margin-top:-6px;margin-left:-5px;border-right-color:#333}#tiptip_holder.tip_left{padding-right:5px}#tiptip_holder.tip_left #tiptip_arrow_inner{margin-top:-6px;margin-left:-7px;border-left-color:#333}#tiptip_content{color:#fff;font-size:.8em;max-width:150px;background:#333;text-align:center;border-radius:3px;padding:.618em 1em;box-shadow:0 1px 3px rgba(0,0,0,.2)}#tiptip_content code{padding:1px;background:#888}#tiptip_arrow,#tiptip_arrow_inner{position:absolute;border-color:transparent;border-style:solid;border-width:6px;height:0;width:0}.dhl-preferred-service-content{margin-top:1em}.dhl-preferred-service-content .dhl-hidden{display:none}.dhl-preferred-service-content .dhl-preferred-service-cost{font-size:.9em}.dhl-preferred-service-content .dhl-preferred-service-item{margin-bottom:1em}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-logo{margin-bottom:1em}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-logo img{margin:0;padding:0;max-height:100px;max-width:100px;background:#fc0}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-title{font-weight:700;font-size:1em;margin-bottom:.5em}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-cost{margin-bottom:.5em}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-desc{font-size:.9em;margin-bottom:.5em}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times{display:flex;flex-direction:row;flex-wrap:wrap;margin:0;padding:0}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li{flex-basis:10%;display:inline-block;text-align:center;padding:10px 0 0;margin:0 8px 8px 0;background-color:#e3e3e3}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li label{position:relative;display:flex;flex-direction:column;flex-wrap:wrap;padding:5px 10px;font-size:.9em;font-weight:700;background-color:#eef4f2;cursor:pointer;margin:0;color:#5f7285}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li label .dhl-preferred-time-title{font-size:1.2em}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li input[type=radio]{opacity:0;width:1px;height:1px;position:absolute}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times li input[type=radio]:checked~label{background-color:#fc0}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times.dhl-preferred-service-time li{flex-grow:inherit;flex-basis:inherit}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-times.dhl-preferred-service-time li label .dhl-preferred-time-title{font-size:1em}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-service-data input[type=text]{width:100%;margin-bottom:.5em}.dhl-preferred-service-content .dhl-preferred-service-item .woocommerce-help-tip{background:#fc0;display:inline-block;font-size:1em;font-style:normal;height:18px;padding:3px;margin-top:-5px;margin-left:5px;border-radius:50%;line-height:18px;position:relative;vertical-align:middle;width:18px}.dhl-preferred-service-content .dhl-preferred-service-item .woocommerce-help-tip::after{position:absolute;top:0;left:0;width:100%;height:100%;text-align:center;cursor:help;content:"?"}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-delivery-types,.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-location-types{list-style:none;margin:0;margin-bottom:1em;display:flex;flex-wrap:wrap;justify-content:flex-start;align-items:center}.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-delivery-types li,.dhl-preferred-service-content .dhl-preferred-service-item .dhl-preferred-location-types li{margin-right:1em}.dhl-preferred-service-content .dhl-preferred-service-item.dhl-preferred-service-header .dhl-preferred-service-title{font-size:1.1em} \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/css/preferred-services.scss b/packages/woocommerce-germanized-dhl/assets/css/preferred-services.scss new file mode 100644 index 000000000..72123a694 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/css/preferred-services.scss @@ -0,0 +1,230 @@ +#tiptip_holder { + display: none; + z-index: 8675309; + position: absolute; + top: 0; + + /*rtl:ignore*/ + left: 0; + + &.tip_top { + padding-bottom: 5px; + + #tiptip_arrow_inner { + margin-top: -7px; + margin-left: -6px; + border-top-color: #333; + } + } + + &.tip_bottom { + padding-top: 5px; + + #tiptip_arrow_inner { + margin-top: -5px; + margin-left: -6px; + border-bottom-color: #333; + } + } + + &.tip_right { + padding-left: 5px; + + #tiptip_arrow_inner { + margin-top: -6px; + margin-left: -5px; + border-right-color: #333; + } + } + + &.tip_left { + padding-right: 5px; + + #tiptip_arrow_inner { + margin-top: -6px; + margin-left: -7px; + border-left-color: #333; + } + } +} + +#tiptip_content { + color: #fff; + font-size: 0.8em; + max-width: 150px; + background: #333; + text-align: center; + border-radius: 3px; + padding: 0.618em 1em; + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2); + + code { + padding: 1px; + background: #888; + } +} + +#tiptip_arrow, +#tiptip_arrow_inner { + position: absolute; + border-color: transparent; + border-style: solid; + border-width: 6px; + height: 0; + width: 0; +} + +.dhl-preferred-service-content { + margin-top: 1em; + + .dhl-hidden { + display: none; + } + + .dhl-preferred-service-cost { + font-size: .9em; + } + + .dhl-preferred-service-item { + margin-bottom: 1em; + + .dhl-preferred-service-logo { + img { + margin: 0; + padding: 0; + max-height: 100px; + max-width: 100px; + background: #FFCC00; + } + margin-bottom: 1em; + } + + .dhl-preferred-service-title { + font-weight: bold; + font-size: 1em; + margin-bottom: .5em; + } + + .dhl-preferred-service-cost { + margin-bottom: .5em; + } + + .dhl-preferred-service-desc { + font-size: .9em; + margin-bottom: .5em; + } + + .dhl-preferred-service-times { + display: flex; + flex-direction: row; + flex-wrap: wrap; + margin: 0; + padding: 0; + + li { + flex-basis: 10%; + display: inline-block; + text-align: center; + padding: 10px 0 0; + margin: 0 8px 8px 0; + background-color: #e3e3e3; + + label { + position: relative; + display: flex; + flex-direction: column; + flex-wrap: wrap; + padding: 5px 10px; + font-size: .9em; + font-weight: bold; + background-color: #eef4f2; + cursor: pointer; + margin: 0; + color: #5f7285; + + .dhl-preferred-time-title { + font-size: 1.2em; + } + } + + input[type=radio] { + opacity: 0; + width: 1px; + height: 1px; + position: absolute; + + &:checked ~ label { + background-color: #FFCC00; + } + } + } + + &.dhl-preferred-service-time { + li { + flex-grow: inherit; + flex-basis: inherit; + + label { + .dhl-preferred-time-title { + font-size: 1em; + } + } + } + } + } + + .dhl-preferred-service-data { + input[type=text] { + width: 100%; + margin-bottom: .5em; + } + } + + .woocommerce-help-tip { + background: #ffcc00; + display: inline-block; + font-size: 1em; + font-style: normal; + height: 18px; + padding: 3px; + margin-top: -5px; + margin-left: 5px; + border-radius: 50%; + line-height: 18px; + position: relative; + vertical-align: middle; + width: 18px; + + &::after { + position: absolute; + top: 0; + left: 0; + width: 100%; + height: 100%; + text-align: center; + cursor: help; + content: "?"; + } + } + + .dhl-preferred-location-types, .dhl-preferred-delivery-types { + list-style: none; + margin: 0; + margin-bottom: 1em; + display: flex; + flex-wrap: wrap; + justify-content: flex-start; + align-items: center; + + li { + margin-right: 1em; + } + } + + &.dhl-preferred-service-header { + .dhl-preferred-service-title { + font-size: 1.1em; + } + } + } +} \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/img/dhl-official.png b/packages/woocommerce-germanized-dhl/assets/img/dhl-official.png new file mode 100755 index 000000000..18dab3cf6 Binary files /dev/null and b/packages/woocommerce-germanized-dhl/assets/img/dhl-official.png differ diff --git a/packages/woocommerce-germanized-dhl/assets/img/packstation.png b/packages/woocommerce-germanized-dhl/assets/img/packstation.png new file mode 100755 index 000000000..903d8388b Binary files /dev/null and b/packages/woocommerce-germanized-dhl/assets/img/packstation.png differ diff --git a/packages/woocommerce-germanized-dhl/assets/img/parcelshop.png b/packages/woocommerce-germanized-dhl/assets/img/parcelshop.png new file mode 100755 index 000000000..15fafae42 Binary files /dev/null and b/packages/woocommerce-germanized-dhl/assets/img/parcelshop.png differ diff --git a/packages/woocommerce-germanized-dhl/assets/img/post_office.png b/packages/woocommerce-germanized-dhl/assets/img/post_office.png new file mode 100755 index 000000000..097023dcd Binary files /dev/null and b/packages/woocommerce-germanized-dhl/assets/img/post_office.png differ diff --git a/packages/woocommerce-germanized-dhl/assets/img/wp-int-eu-preview.png b/packages/woocommerce-germanized-dhl/assets/img/wp-int-eu-preview.png new file mode 100644 index 000000000..a4438cd37 Binary files /dev/null and b/packages/woocommerce-germanized-dhl/assets/img/wp-int-eu-preview.png differ diff --git a/packages/woocommerce-germanized-dhl/assets/img/wp-int-preview.png b/packages/woocommerce-germanized-dhl/assets/img/wp-int-preview.png new file mode 100644 index 000000000..6ac0f7e34 Binary files /dev/null and b/packages/woocommerce-germanized-dhl/assets/img/wp-int-preview.png differ diff --git a/packages/woocommerce-germanized-dhl/assets/js/admin-deutsche-post-label.js b/packages/woocommerce-germanized-dhl/assets/js/admin-deutsche-post-label.js new file mode 100644 index 000000000..2bfd715a3 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/js/admin-deutsche-post-label.js @@ -0,0 +1,110 @@ +window.germanized = window.germanized || {}; +window.germanized.admin = window.germanized.admin || {}; + +( function( $, admin ) { + + /** + * Core + */ + admin.dhl_post_label = { + + params: {}, + + init: function () { + var self = admin.dhl_post_label; + self.params = wc_gzd_admin_deutsche_post_label_params; + + $( document ).on( 'change', '#wc-gzd-shipment-label-admin-fields-deutsche_post #product_id, #wc-gzd-shipment-label-admin-fields-deutsche_post #wc-gzd-shipment-label-wrapper-additional-services :input', self.onRefreshPreview ); + }, + + getSelectedAdditionalServices: function() { + var selectedIds = $( "#wc-gzd-shipment-label-wrapper-additional-services :input:checked" ).map( function() { + return $( this ).attr( 'name' ).replace( 'service_', '' ); + }).get(); + + return selectedIds; + }, + + onRefreshPreview: function() { + var self = admin.dhl_post_label, + backbone = germanized.admin.shipment_label_backbone.backbone, + params = {}, + $wrapper = $( '.wc-gzd-shipment-create-label' ); + + params['security'] = self.params.refresh_label_preview_nonce; + params['product_id'] = self.getProductId(); + params['selected_services'] = self.getSelectedAdditionalServices(); + params['action'] = 'woocommerce_gzd_dhl_refresh_deutsche_post_label_preview'; + + backbone.doAjax( params, $wrapper, self.onPreviewSuccess ); + }, + + onPreviewSuccess: function( data ) { + var self = admin.dhl_post_label, + $wrapper = $( '.wc-gzd-dhl-im-product-data .col-preview' ), + $img_wrapper = $( '.wc-gzd-dhl-im-product-data' ).find( '.image-preview' ); + + if ( data.is_wp_int ) { + $wrapper.parents( '.wc-gzd-shipment-create-label' ).find( '.page_format_field' ).hide(); + } else { + $wrapper.parents( '.wc-gzd-shipment-create-label' ).find( '.page_format_field' ).show(); + } + + if ( data.preview_url ) { + $wrapper.block({ + message: null, + overlayCSS: { + background: '#fff', + opacity: 0.6 + } + }); + + if ( $img_wrapper.find( '.stamp-preview' ).length <= 0 ) { + $img_wrapper.append( '' ); + } + + self.replaceProductData( data.preview_data ); + + $img_wrapper.find( '.stamp-preview' ).attr('src', data.preview_url ).load( function() { + $wrapper.unblock(); + $( this ).show(); + }); + } else { + $img_wrapper.html( '' ); + } + }, + + refreshProductData: function() { + var self = admin.dhl_post_label; + + self.onRefreshPreview(); + }, + + getProductId: function() { + return $( '#wc-gzd-shipment-label-admin-fields-deutsche_post #product_id' ).val(); + }, + + replaceProductData: function( productData ) { + var self = admin.dhl_post_label, + $wrapper = $( '.wc-gzd-shipment-create-label' ).find( '.wc-gzd-dhl-im-product-data' ); + + $wrapper.find( '.data-placeholder' ).html( '' ); + + $wrapper.find( '.data-placeholder' ).each( function() { + var replaceKey = $( this ).data( 'replace' ); + + if ( productData.hasOwnProperty( replaceKey ) ) { + $( this ).html( productData[ replaceKey ] ); + $( this ).show(); + } else { + $( this ).hide(); + } + } ); + } + }; + + $( document ).ready( function() { + germanized.admin.dhl_post_label.init(); + }); + +})( jQuery, window.germanized.admin ); \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/js/admin-deutsche-post-label.min.js b/packages/woocommerce-germanized-dhl/assets/js/admin-deutsche-post-label.min.js new file mode 100644 index 000000000..9a979c4b6 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/js/admin-deutsche-post-label.min.js @@ -0,0 +1 @@ +window.germanized=window.germanized||{},window.germanized.admin=window.germanized.admin||{},function(d,n){n.dhl_post_label={params:{},init:function(){var e=n.dhl_post_label;e.params=wc_gzd_admin_deutsche_post_label_params,d(document).on("change","#wc-gzd-shipment-label-admin-fields-deutsche_post #product_id, #wc-gzd-shipment-label-admin-fields-deutsche_post #wc-gzd-shipment-label-wrapper-additional-services :input",e.onRefreshPreview)},getSelectedAdditionalServices:function(){return d("#wc-gzd-shipment-label-wrapper-additional-services :input:checked").map(function(){return d(this).attr("name").replace("service_","")}).get()},onRefreshPreview:function(){var e=n.dhl_post_label,a=germanized.admin.shipment_label_backbone.backbone,i={},t=d(".wc-gzd-shipment-create-label");i.security=e.params.refresh_label_preview_nonce,i.product_id=e.getProductId(),i.selected_services=e.getSelectedAdditionalServices(),i.action="woocommerce_gzd_dhl_refresh_deutsche_post_label_preview",a.doAjax(i,t,e.onPreviewSuccess)},onPreviewSuccess:function(e){var a=n.dhl_post_label,i=d(".wc-gzd-dhl-im-product-data .col-preview"),t=d(".wc-gzd-dhl-im-product-data").find(".image-preview");e.is_wp_int?i.parents(".wc-gzd-shipment-create-label").find(".page_format_field").hide():i.parents(".wc-gzd-shipment-create-label").find(".page_format_field").show(),e.preview_url?(i.block({message:null,overlayCSS:{background:"#fff",opacity:.6}}),t.find(".stamp-preview").length<=0&&t.append(''),a.replaceProductData(e.preview_data),t.find(".stamp-preview").attr("src",e.preview_url).load(function(){i.unblock(),d(this).show()})):t.html("")},refreshProductData:function(){n.dhl_post_label.onRefreshPreview()},getProductId:function(){return d("#wc-gzd-shipment-label-admin-fields-deutsche_post #product_id").val()},replaceProductData:function(a){n.dhl_post_label;var e=d(".wc-gzd-shipment-create-label").find(".wc-gzd-dhl-im-product-data");e.find(".data-placeholder").html(""),e.find(".data-placeholder").each(function(){var e=d(this).data("replace");a.hasOwnProperty(e)?(d(this).html(a[e]),d(this).show()):d(this).hide()})}},d(document).ready(function(){germanized.admin.dhl_post_label.init()})}(jQuery,window.germanized.admin); \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/js/admin-internetmarke.js b/packages/woocommerce-germanized-dhl/assets/js/admin-internetmarke.js new file mode 100644 index 000000000..cf3528bfc --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/js/admin-internetmarke.js @@ -0,0 +1,266 @@ +window.germanized = window.germanized || {}; +window.germanized.admin = window.germanized.admin || {}; + +/* + * http://www.myersdaily.org/joseph/javascript/md5-text.html + */ +( function (global) { + + var md5cycle = function (x, k) { + var a = x[0], + b = x[1], + c = x[2], + d = x[3]; + + a = ff(a, b, c, d, k[0], 7, -680876936); + d = ff(d, a, b, c, k[1], 12, -389564586); + c = ff(c, d, a, b, k[2], 17, 606105819); + b = ff(b, c, d, a, k[3], 22, -1044525330); + a = ff(a, b, c, d, k[4], 7, -176418897); + d = ff(d, a, b, c, k[5], 12, 1200080426); + c = ff(c, d, a, b, k[6], 17, -1473231341); + b = ff(b, c, d, a, k[7], 22, -45705983); + a = ff(a, b, c, d, k[8], 7, 1770035416); + d = ff(d, a, b, c, k[9], 12, -1958414417); + c = ff(c, d, a, b, k[10], 17, -42063); + b = ff(b, c, d, a, k[11], 22, -1990404162); + a = ff(a, b, c, d, k[12], 7, 1804603682); + d = ff(d, a, b, c, k[13], 12, -40341101); + c = ff(c, d, a, b, k[14], 17, -1502002290); + b = ff(b, c, d, a, k[15], 22, 1236535329); + + a = gg(a, b, c, d, k[1], 5, -165796510); + d = gg(d, a, b, c, k[6], 9, -1069501632); + c = gg(c, d, a, b, k[11], 14, 643717713); + b = gg(b, c, d, a, k[0], 20, -373897302); + a = gg(a, b, c, d, k[5], 5, -701558691); + d = gg(d, a, b, c, k[10], 9, 38016083); + c = gg(c, d, a, b, k[15], 14, -660478335); + b = gg(b, c, d, a, k[4], 20, -405537848); + a = gg(a, b, c, d, k[9], 5, 568446438); + d = gg(d, a, b, c, k[14], 9, -1019803690); + c = gg(c, d, a, b, k[3], 14, -187363961); + b = gg(b, c, d, a, k[8], 20, 1163531501); + a = gg(a, b, c, d, k[13], 5, -1444681467); + d = gg(d, a, b, c, k[2], 9, -51403784); + c = gg(c, d, a, b, k[7], 14, 1735328473); + b = gg(b, c, d, a, k[12], 20, -1926607734); + + a = hh(a, b, c, d, k[5], 4, -378558); + d = hh(d, a, b, c, k[8], 11, -2022574463); + c = hh(c, d, a, b, k[11], 16, 1839030562); + b = hh(b, c, d, a, k[14], 23, -35309556); + a = hh(a, b, c, d, k[1], 4, -1530992060); + d = hh(d, a, b, c, k[4], 11, 1272893353); + c = hh(c, d, a, b, k[7], 16, -155497632); + b = hh(b, c, d, a, k[10], 23, -1094730640); + a = hh(a, b, c, d, k[13], 4, 681279174); + d = hh(d, a, b, c, k[0], 11, -358537222); + c = hh(c, d, a, b, k[3], 16, -722521979); + b = hh(b, c, d, a, k[6], 23, 76029189); + a = hh(a, b, c, d, k[9], 4, -640364487); + d = hh(d, a, b, c, k[12], 11, -421815835); + c = hh(c, d, a, b, k[15], 16, 530742520); + b = hh(b, c, d, a, k[2], 23, -995338651); + + a = ii(a, b, c, d, k[0], 6, -198630844); + d = ii(d, a, b, c, k[7], 10, 1126891415); + c = ii(c, d, a, b, k[14], 15, -1416354905); + b = ii(b, c, d, a, k[5], 21, -57434055); + a = ii(a, b, c, d, k[12], 6, 1700485571); + d = ii(d, a, b, c, k[3], 10, -1894986606); + c = ii(c, d, a, b, k[10], 15, -1051523); + b = ii(b, c, d, a, k[1], 21, -2054922799); + a = ii(a, b, c, d, k[8], 6, 1873313359); + d = ii(d, a, b, c, k[15], 10, -30611744); + c = ii(c, d, a, b, k[6], 15, -1560198380); + b = ii(b, c, d, a, k[13], 21, 1309151649); + a = ii(a, b, c, d, k[4], 6, -145523070); + d = ii(d, a, b, c, k[11], 10, -1120210379); + c = ii(c, d, a, b, k[2], 15, 718787259); + b = ii(b, c, d, a, k[9], 21, -343485551); + + x[0] = add32(a, x[0]); + x[1] = add32(b, x[1]); + x[2] = add32(c, x[2]); + x[3] = add32(d, x[3]); + + } + + var cmn = function (q, a, b, x, s, t) { + a = add32(add32(a, q), add32(x, t)); + return add32((a << s) | (a >>> (32 - s)), b); + } + + var ff = function (a, b, c, d, x, s, t) { + return cmn((b & c) | ((~b) & d), a, b, x, s, t); + } + + var gg = function (a, b, c, d, x, s, t) { + return cmn((b & d) | (c & (~d)), a, b, x, s, t); + } + + var hh = function (a, b, c, d, x, s, t) { + return cmn(b ^ c ^ d, a, b, x, s, t); + } + + var ii = function (a, b, c, d, x, s, t) { + return cmn(c ^ (b | (~d)), a, b, x, s, t); + } + + var md51 = function (s) { + var txt = '', + n = s.length, + state = [1732584193, -271733879, -1732584194, 271733878], + i; + for (i = 64; i <= s.length; i += 64) { + md5cycle(state, md5blk(s.substring(i - 64, i))); + } + s = s.substring(i - 64); + var tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; + for (i = 0; i < s.length; i++) + tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3); + tail[i >> 2] |= 0x80 << ((i % 4) << 3); + if (i > 55) { + md5cycle(state, tail); + for (i = 0; i < 16; i++) tail[i] = 0; + } + tail[14] = n * 8; + md5cycle(state, tail); + return state; + } + + /* there needs to be support for Unicode here, + * unless we pretend that we can redefine the MD-5 + * algorithm for multi-byte characters (perhaps + * by adding every four 16-bit characters and + * shortening the sum to 32 bits). Otherwise + * I suggest performing MD-5 as if every character + * was two bytes--e.g., 0040 0025 = @%--but then + * how will an ordinary MD-5 sum be matched? + * There is no way to standardize text to something + * like UTF-8 before transformation; speed cost is + * utterly prohibitive. The JavaScript standard + * itself needs to look at this: it should start + * providing access to strings as preformed UTF-8 + * 8-bit unsigned value arrays. + */ + var md5blk = function (s) { /* I figured global was faster. */ + var md5blks = [], + i; /* Andy King said do it this way. */ + for (i = 0; i < 64; i += 4) { + md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) << 24); + } + return md5blks; + } + + var hex_chr = '0123456789abcdef'.split(''); + + var rhex = function (n) { + var s = '', + j = 0; + for (; j < 4; j++) + s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F]; + return s; + } + + var hex = function (x) { + for (var i = 0; i < x.length; i++) + x[i] = rhex(x[i]); + return x.join(''); + } + + var md5 = global.md5 = function (s) { + return hex(md51(s)); + } + + /* this function is much faster, + so if possible we use it. Some IEs + are the only ones I know of that + need the idiotic second function, + generated by an if clause. */ + + var add32 = function (a, b) { + return (a + b) & 0xFFFFFFFF; + } + + if (md5('hello') != '5d41402abc4b2a76b9719d911017c592') { + var add32 = function (x, y) { + var lsw = (x & 0xFFFF) + (y & 0xFFFF), + msw = (x >> 16) + (y >> 16) + (lsw >> 16); + return (msw << 16) | (lsw & 0xFFFF); + } + } +})(window); + +( function( $, admin, window ) { + + /** + * Core + */ + admin.dhl_internetmarke = { + + params: {}, + + init: function () { + var self = admin.dhl_internetmarke; + + $( document ).on( 'click', '#woocommerce_gzd_dhl_im_portokasse_charge', self.onCharge ); + }, + + onCharge: function() { + var $button = $( this ), + data = $button.data(), + amount = $( '#woocommerce_gzd_dhl_im_portokasse_charge_amount' ).val(); + + $form = $( '
' ).appendTo( 'body' ); + + $.each( data, function( index, value ) { + $form.append( '' ); + } ); + + var balance = parseInt( ( parseFloat( amount.replace( ',', '.') ).toFixed( 2 ) * 100 ).toFixed() ); + var wallet = parseInt( data['wallet'] ); + + /** + * Set min amount. + */ + if ( balance < 1000 || Number.isNaN( balance ) ) { + balance = 1000; + } + + var date = new Date(); + var timestamp = + ('0' + date.getDate()).slice(-2) + + ('0' + (date.getMonth() + 1)).slice(-2) + + date.getFullYear().toString() + + '-' + + ('0' + date.getHours()).slice(-2) + + ('0' + date.getMinutes()).slice(-2) + + ('0' + date.getSeconds()).slice(-2); + + var concat = [ + data['partner_id'], + timestamp, + data['success_url'], + data['cancel_url'], + data['user_token'], + wallet + balance, + data['schluessel_dpwn_partner'] + ].join( '::' ); + + $form.append( '' ); + $form.append( '' ); + $form.append( '' ); + + $form.submit(); + + return false; + } + }; + + $( document ).ready( function() { + germanized.admin.dhl_internetmarke.init(); + }); + +})( jQuery, window.germanized.admin, window ); \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/js/admin-internetmarke.min.js b/packages/woocommerce-germanized-dhl/assets/js/admin-internetmarke.min.js new file mode 100644 index 000000000..ff0186a70 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/js/admin-internetmarke.min.js @@ -0,0 +1 @@ +window.germanized=window.germanized||{},window.germanized.admin=window.germanized.admin||{},function(e){var a=function(e,n){var r=e[0],t=e[1],o=e[2],a=e[3],r=i(r,t,o,a,n[0],7,-680876936),a=i(a,r,t,o,n[1],12,-389564586),o=i(o,a,r,t,n[2],17,606105819),t=i(t,o,a,r,n[3],22,-1044525330);r=i(r,t,o,a,n[4],7,-176418897),a=i(a,r,t,o,n[5],12,1200080426),o=i(o,a,r,t,n[6],17,-1473231341),t=i(t,o,a,r,n[7],22,-45705983),r=i(r,t,o,a,n[8],7,1770035416),a=i(a,r,t,o,n[9],12,-1958414417),o=i(o,a,r,t,n[10],17,-42063),t=i(t,o,a,r,n[11],22,-1990404162),r=i(r,t,o,a,n[12],7,1804603682),a=i(a,r,t,o,n[13],12,-40341101),o=i(o,a,r,t,n[14],17,-1502002290),t=i(t,o,a,r,n[15],22,1236535329),r=u(r,t,o,a,n[1],5,-165796510),a=u(a,r,t,o,n[6],9,-1069501632),o=u(o,a,r,t,n[11],14,643717713),t=u(t,o,a,r,n[0],20,-373897302),r=u(r,t,o,a,n[5],5,-701558691),a=u(a,r,t,o,n[10],9,38016083),o=u(o,a,r,t,n[15],14,-660478335),t=u(t,o,a,r,n[4],20,-405537848),r=u(r,t,o,a,n[9],5,568446438),a=u(a,r,t,o,n[14],9,-1019803690),o=u(o,a,r,t,n[3],14,-187363961),t=u(t,o,a,r,n[8],20,1163531501),r=u(r,t,o,a,n[13],5,-1444681467),a=u(a,r,t,o,n[2],9,-51403784),o=u(o,a,r,t,n[7],14,1735328473),t=u(t,o,a,r,n[12],20,-1926607734),r=c(r,t,o,a,n[5],4,-378558),a=c(a,r,t,o,n[8],11,-2022574463),o=c(o,a,r,t,n[11],16,1839030562),t=c(t,o,a,r,n[14],23,-35309556),r=c(r,t,o,a,n[1],4,-1530992060),a=c(a,r,t,o,n[4],11,1272893353),o=c(o,a,r,t,n[7],16,-155497632),t=c(t,o,a,r,n[10],23,-1094730640),r=c(r,t,o,a,n[13],4,681279174),a=c(a,r,t,o,n[0],11,-358537222),o=c(o,a,r,t,n[3],16,-722521979),t=c(t,o,a,r,n[6],23,76029189),r=c(r,t,o,a,n[9],4,-640364487),a=c(a,r,t,o,n[12],11,-421815835),o=c(o,a,r,t,n[15],16,530742520),t=c(t,o,a,r,n[2],23,-995338651),r=m(r,t,o,a,n[0],6,-198630844),a=m(a,r,t,o,n[7],10,1126891415),o=m(o,a,r,t,n[14],15,-1416354905),t=m(t,o,a,r,n[5],21,-57434055),r=m(r,t,o,a,n[12],6,1700485571),a=m(a,r,t,o,n[3],10,-1894986606),o=m(o,a,r,t,n[10],15,-1051523),t=m(t,o,a,r,n[1],21,-2054922799),r=m(r,t,o,a,n[8],6,1873313359),a=m(a,r,t,o,n[15],10,-30611744),o=m(o,a,r,t,n[6],15,-1560198380),t=m(t,o,a,r,n[13],21,1309151649),r=m(r,t,o,a,n[4],6,-145523070),a=m(a,r,t,o,n[11],10,-1120210379),o=m(o,a,r,t,n[2],15,718787259),t=m(t,o,a,r,n[9],21,-343485551),e[0]=l(r,e[0]),e[1]=l(t,e[1]),e[2]=l(o,e[2]),e[3]=l(a,e[3])},d=function(e,n,r,t,o,a){return n=l(l(n,e),l(t,a)),l(n<>>32-o,r)},i=function(e,n,r,t,o,a,i){return d(n&r|~n&t,e,n,o,a,i)},u=function(e,n,r,t,o,a,i){return d(n&t|r&~t,e,n,o,a,i)},c=function(e,n,r,t,o,a,i){return d(n^r^t,e,n,o,a,i)},m=function(e,n,r,t,o,a,i){return d(r^(n|~t),e,n,o,a,i)},t="0123456789abcdef".split(""),n=function(e){for(var n=0;n>8*r+4&15]+t[e>>8*r&15];return n}(e[n]);return e.join("")},e=e.md5=function(e){return n(function(e){for(var n=e.length,r=[1732584193,-271733879,-1732584194,271733878],t=64;t<=e.length;t+=64)a(r,function(e){for(var n=[],r,r=0;r<64;r+=4)n[r>>2]=e.charCodeAt(r)+(e.charCodeAt(r+1)<<8)+(e.charCodeAt(r+2)<<16)+(e.charCodeAt(r+3)<<24);return n}(e.substring(t-64,t)));e=e.substring(t-64);var o=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];for(t=0;t>2]|=e.charCodeAt(t)<<(t%4<<3);if(o[t>>2]|=128<<(t%4<<3),55>16)+(n>>16)+(r>>16)<<16|65535&r})}(window),function(o,n,a){n.dhl_internetmarke={params:{},init:function(){var e=n.dhl_internetmarke;o(document).on("click","#woocommerce_gzd_dhl_im_portokasse_charge",e.onCharge)},onCharge:function(){var e=o(this),n=e.data(),r=o("#woocommerce_gzd_dhl_im_portokasse_charge_amount").val();$form=o('
').appendTo("body"),o.each(n,function(e,n){$form.append('')});var t=parseInt((100*parseFloat(r.replace(",",".")).toFixed(2)).toFixed()),e=parseInt(n.wallet);(t<1e3||Number.isNaN(t))&&(t=1e3);r=new Date,r=("0"+r.getDate()).slice(-2)+("0"+(r.getMonth()+1)).slice(-2)+r.getFullYear().toString()+"-"+("0"+r.getHours()).slice(-2)+("0"+r.getMinutes()).slice(-2)+("0"+r.getSeconds()).slice(-2),n=[n.partner_id,r,n.success_url,n.cancel_url,n.user_token,e+t,n.schluessel_dpwn_partner].join("::");return $form.append(''),$form.append(''),$form.append(''),$form.submit(),!1}},o(document).ready(function(){germanized.admin.dhl_internetmarke.init()})}(jQuery,window.germanized.admin,window); \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/js/parcel-finder.js b/packages/woocommerce-germanized-dhl/assets/js/parcel-finder.js new file mode 100644 index 000000000..a4dfc1f0f --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/js/parcel-finder.js @@ -0,0 +1,309 @@ +window.germanized = window.germanized || {}; +window.germanized.dhl_parcel_finder = window.germanized.dhl_parcel_finder || {}; + +( function( $, germanized ) { + + /** + * Core + */ + germanized.dhl_parcel_finder = { + + params: {}, + parcelShops: [], + wrapper: '', + + init: function () { + var self = germanized.dhl_parcel_finder; + self.params = wc_gzd_dhl_parcel_finder_params; + self.wrapper = self.params.wrapper; + + $( document ) + .on( 'click', '.gzd-dhl-parcel-shop-modal', self.openModal ) + .on( 'click', '#dhl-parcel-finder-wrapper .dhl-parcel-finder-close', self.closeModal ) + .on( 'submit', '#dhl-parcel-finder-wrapper #dhl-parcel-finder-form', self.onSubmit ) + .on( 'click', '#dhl-parcel-finder-wrapper .dhl-retry-search', self.onSubmit ) + .on( 'click', '#dhl-parcel-finder-wrapper .dhl-parcelshop-select-btn', self.onSelectShop ); + + $( document.body ).on( 'woocommerce_gzd_dhl_location_available_pickup_types_changed', self.onChangeAvailablePickupTypes ); + }, + + onChangeAvailablePickupTypes: function() { + var self = germanized.dhl_parcel_finder, + loc = germanized.dhl_parcel_locator, + $modal = self.getModal(), + method = loc.getShippingMethod(), + methodData = loc.getShippingMethodData( method ); + + if ( methodData ) { + + $modal.find( '.finder-pickup-type' ).addClass( 'hidden' ); + + $.each( methodData.supports, function( i, pickupType ) { + var $type = $modal.find( '.finder-pickup-type[data-pickup_type="' + pickupType + '"]' ); + + $type.find( 'input[type=checkbox]' ).prop( 'checked', true ); + $type.removeClass( 'hidden' ); + }); + } + }, + + openModal: function() { + var self = germanized.dhl_parcel_finder, + $modal = self.getModal(); + + var country = $( self.wrapper + ' #shipping_country' ).val().length > 0 ? $( self.wrapper + ' #shipping_country' ).val() : $( self.wrapper + ' #billing_country' ).val(); + $modal.find( '#dhl-parcelfinder-country').val( country ); + + var postcode = $( self.wrapper + ' #shipping_postcode' ).val().length > 0 ? $( self.wrapper + ' #shipping_postcode' ).val() : $( self.wrapper + ' #billing_postcode' ).val(); + $modal.find( '#dhl-parcelfinder-postcode' ).val( postcode ); + + var city = $( self.wrapper + ' #shipping_city' ).val().length > 0 ? $( self.wrapper + ' #shipping_city' ).val() : $( self.wrapper + ' #billing_city' ).val(); + $modal.find( '#dhl-parcelfinder-city' ).val( city ); + + $modal.addClass( 'open' ); + $modal.find( '#dhl-parcel-finder-form' ).submit(); + + return false; + }, + + closeModal: function() { + var self = germanized.dhl_parcel_finder; + self.getModal().removeClass( 'open' ); + + return false; + }, + + getModal: function() { + return $( '#dhl-parcel-finder-wrapper' ); + }, + + doAjax: function( params, $wrapper, cSuccess, cError ) { + var self = germanized.dhl_parcel_finder; + + cSuccess = cSuccess || self.onAjaxSuccess; + cError = cError || self.onAjaxError; + + if ( ! params.hasOwnProperty( 'security' ) ) { + params['security'] = self.params.parcel_finder_nonce; + } + + $wrapper.find( '#dhl-parcel-finder-map' ).block({ + message: null, + overlayCSS: { + background: '#fff', + opacity: 0.6 + } + }); + + $wrapper.find( '.notice-wrapper' ).empty(); + + $.ajax({ + type: "POST", + url: self.params.ajax_url, + data: params, + success: function( data ) { + if ( data.success ) { + $wrapper.find( '#dhl-parcel-finder-map' ).unblock(); + cSuccess.apply( $wrapper, [ data ] ); + } else { + cError.apply( $wrapper, [ data ] ); + $wrapper.find( '#dhl-parcel-finder-map' ).unblock(); + + if ( data.hasOwnProperty( 'message' ) ) { + self.addNotice( data.message, 'error', $wrapper ); + } else if( data.hasOwnProperty( 'messages' ) ) { + $.each( data.messages, function( i, message ) { + self.addNotice( message, 'error', $wrapper ); + }); + } + } + }, + error: function( data ) {}, + dataType: 'json' + }); + }, + + onAjaxSuccess: function( data ) {}, + + onAjaxError: function( data ) {}, + + getFormData: function( $form ) { + var data = {}; + + $.each( $form.serializeArray(), function( index, item ) { + if ( item.name.indexOf( '[]' ) !== -1 ) { + item.name = item.name.replace( '[]', '' ); + data[ item.name ] = $.makeArray( data[ item.name ] ); + data[ item.name ].push( item.value ); + } else { + data[ item.name ] = item.value; + } + }); + + return data; + }, + + onSubmit: function( e ) { + var self = germanized.dhl_parcel_finder, + loc = germanized.dhl_parcel_locator, + $modal = self.getModal(), + $content = $modal.find( '#dhl-parcel-finder' ), + $form = $content.find( 'form' ), + params = self.getFormData( $form ); + + params['action'] = 'woocommerce_gzd_dhl_parcelfinder_search'; + params['is_checkout'] = loc.isCheckout() ? 'yes' : 'no'; + + self.doAjax( params, $content, self.onSubmitSuccess ); + + return false; + }, + + onSubmitSuccess: function( data ) { + var self = germanized.dhl_parcel_finder; + + if ( data.parcel_shops ) { + self.parcelShops = data.parcel_shops; + + if ( typeof google === 'object' && typeof google.maps === 'object' ) { + self.updateMap(); + } else { + self.loadMapsAPI(); + } + } + }, + + loadMapsAPI: function() { + var self = germanized.dhl_parcel_finder; + + self.addScript( 'https://maps.googleapis.com/maps/api/js?key=' + self.params.api_key, self.updateMap ); + }, + + addScript: function( url, callback ) { + var script = document.createElement( 'script' ); + + if ( callback ) { + script.onload = callback; + } + + script.type = 'text/javascript'; + script.src = url; + document.body.appendChild( script ); + }, + + updateMap: function() { + var self = germanized.dhl_parcel_finder, + parcelShops = self.parcelShops; + + var uluru = { + lat: parcelShops[0].place.geo.latitude, + lng: parcelShops[0].place.geo.longitude + }; + + var map = new google.maps.Map( document.getElementById( 'dhl-parcel-finder-map' ), { + zoom: 13, + center: uluru + }); + + var infoWinArray = []; + + $.each( parcelShops, function( key, value ) { + + var uluru = { + lat: value.place.geo.latitude, + lng: value.place.geo.longitude + }; + + var markerIcon = self.params.packstation_icon, + shopLabel = self.params.i18n.packstation; + + switch ( value.gzd_type ) { + case 'parcelshop': + markerIcon = self.params.parcelshop_icon; + shopLabel = self.params.i18n.branch; + break; + case 'postoffice': + markerIcon = self.params.postoffice_icon; + shopLabel = self.params.i18n.branch; + break; + } + + var infowindow = new google.maps.InfoWindow({ + content: value.html_content, + maxWidth: 300 + }); + + infoWinArray.push( infowindow ); + + var marker = new google.maps.Marker({ + position : uluru, + map : map, + title : shopLabel, + animation: google.maps.Animation.DROP, + icon : markerIcon + }); + + marker.addListener('click', function() { + clearOverlays(); + infowindow.open( map, marker ); + }); + }); + + // Clear all info windows + function clearOverlays() { + for ( var i = 0; i < infoWinArray.length; i++ ) { + infoWinArray[i].close(); + } + } + }, + + onSelectShop: function() { + var self = germanized.dhl_parcel_finder, + parcelShopId = $( this ).attr( 'id' ), + $addressType = $( self.wrapper + ' #shipping_address_type' ), + country = $( self.wrapper + ' #shipping_country' ).val(), + fieldKey = germanized.dhl_parcel_locator.getPickupFieldKey( country ); + + $.each( self.parcelShops, function( key, value ) { + if ( value.gzd_result_id === parcelShopId ) { + var isPackstation = 'packstation' === value.gzd_type; + + $( self.wrapper + ' #shipping_first_name' ).val( $( self.wrapper + ' #shipping_first_name' ).val().length > 0 ? $( self.wrapper + ' #shipping_first_name' ).val() : $( self.wrapper + ' #billing_first_name' ).val() ); + $( self.wrapper + ' #shipping_last_name' ).val( $( self.wrapper + ' #shipping_last_name' ).val().length > 0 ? $( self.wrapper + ' #shipping_last_name' ).val() : $( self.wrapper + ' #billing_last_name' ).val() ); + + $( self.wrapper + ' #shipping_' + fieldKey ).val( value.gzd_name ); + + if ( 'DE' === country ) { + $( self.wrapper + ' #shipping_address_2' ).val( '' ); + } else { + $( self.wrapper + ' #shipping_address_1' ).val( value.place.address.streetAddress ); + } + + $( self.wrapper + ' #shipping_postcode' ).val( value.place.address.postalCode ); + $( self.wrapper + ' #shipping_city' ).val( value.place.address.addressLocality ); + + $addressType.val( 'dhl' ).trigger( 'change' ); + + self.closeModal(); + + if ( 'DE' === country ) { + if ( isPackstation && $( self.wrapper + ' #shipping_dhl_postnumber' ).val() === '' ) { + $( self.wrapper + ' #shipping_dhl_postnumber' ).focus(); + } + } + + return true; + } + }); + }, + + addNotice: function( message, noticeType, $wrapper ) { + $wrapper.find( '.notice-wrapper' ).append( '

' + message + '

' ); + } + }; + + $( document ).ready( function() { + germanized.dhl_parcel_finder.init(); + }); + +})( jQuery, window.germanized ); diff --git a/packages/woocommerce-germanized-dhl/assets/js/parcel-finder.min.js b/packages/woocommerce-germanized-dhl/assets/js/parcel-finder.min.js new file mode 100644 index 000000000..d6ee67425 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/js/parcel-finder.min.js @@ -0,0 +1 @@ +window.germanized=window.germanized||{},window.germanized.dhl_parcel_finder=window.germanized.dhl_parcel_finder||{},function(t,s){s.dhl_parcel_finder={params:{},parcelShops:[],wrapper:"",init:function(){var e=s.dhl_parcel_finder;e.params=wc_gzd_dhl_parcel_finder_params,e.wrapper=e.params.wrapper,t(document).on("click",".gzd-dhl-parcel-shop-modal",e.openModal).on("click","#dhl-parcel-finder-wrapper .dhl-parcel-finder-close",e.closeModal).on("submit","#dhl-parcel-finder-wrapper #dhl-parcel-finder-form",e.onSubmit).on("click","#dhl-parcel-finder-wrapper .dhl-retry-search",e.onSubmit).on("click","#dhl-parcel-finder-wrapper .dhl-parcelshop-select-btn",e.onSelectShop),t(document.body).on("woocommerce_gzd_dhl_location_available_pickup_types_changed",e.onChangeAvailablePickupTypes)},onChangeAvailablePickupTypes:function(){var e=s.dhl_parcel_finder,a=s.dhl_parcel_locator,p=e.getModal(),e=a.getShippingMethod(),a=a.getShippingMethodData(e);a&&(p.find(".finder-pickup-type").addClass("hidden"),t.each(a.supports,function(e,a){a=p.find('.finder-pickup-type[data-pickup_type="'+a+'"]');a.find("input[type=checkbox]").prop("checked",!0),a.removeClass("hidden")}))},openModal:function(){var e=s.dhl_parcel_finder,a=e.getModal(),p=(0

'+e+"

")}},t(document).ready(function(){s.dhl_parcel_finder.init()})}(jQuery,window.germanized); \ No newline at end of file diff --git a/packages/woocommerce-germanized-dhl/assets/js/parcel-locator.js b/packages/woocommerce-germanized-dhl/assets/js/parcel-locator.js new file mode 100644 index 000000000..b0dce3202 --- /dev/null +++ b/packages/woocommerce-germanized-dhl/assets/js/parcel-locator.js @@ -0,0 +1,504 @@ + +window.germanized = window.germanized || {}; +window.germanized.dhl_parcel_locator = window.germanized.dhl_parcel_locator || {}; + +( function( $, germanized ) { + + /** + * Core + */ + germanized.dhl_parcel_locator = { + + params: {}, + parcelShops: [], + wrapper: '', + + init: function () { + var self = germanized.dhl_parcel_locator; + self.params = wc_gzd_dhl_parcel_locator_params; + self.wrapper = self.params.wrapper; + + $( document ) + .on( 'change.dhl', self.wrapper + ' #shipping_address_type', self.refreshAddressType ) + .on( 'change.dhl', self.wrapper + ' #shipping_address_1', self.onChangeAddress ) + .on( 'change.dhl', self.wrapper + ' #shipping_address_2', self.onChangeAddress ) + .on( 'change.dhl', self.wrapper + ' #shipping_country', self.onChangeAddress ) + .on( 'change.dhl', self.wrapper + ' #ship-to-different-address-checkbox', self.onChangeShipping ) + .on( 'change.dhl', self.wrapper + ' #shipping_country', self.refreshAvailability ) + .on( 'input.dhl validate.dhl change.dhl', self.wrapper + ' #shipping_dhl_postnumber', self.validatePostnumber ); + + $( document.body ).on( 'payment_method_selected', self.triggerCheckoutRefresh ); + $( document.body ).on( 'updated_checkout', self.afterRefreshCheckout ); + + if ( ! self.isCheckout() ) { + $( document.body ).on( 'country_to_state_changing', function() { + var self = germanized.dhl_parcel_locator; + + setTimeout( function() { + self.refreshAddressType(); + }, 500 ); + } ); + } + + self.refreshAvailability(); + self.refreshAddressType(); + }, + + validatePostnumber: function( e ) { + var $this = $( this ), + $parent = $this.closest( '.form-row' ), + eventType = e.type; + + if ( 'input' === eventType ) { + if ( $this.val() ) { + $this.val( $this.val().replace( /\D/g,'' ) ); + } + } + + if ( 'validate' === eventType || 'change' === eventType ) { + if ( $this.val() ) { + $this.val( $this.val().replace( /\D/g,'' ) ); + + if ( $this.val().toString().length < 6 || $this.val().toString().length > 12 ) { + $parent.removeClass( 'woocommerce-validated' ).addClass( 'woocommerce-invalid woocommerce-invalid-postnumber' ); + } + } + } + }, + + triggerCheckoutRefresh: function() { + $( document.body ).trigger( 'update_checkout' ); + }, + + isCheckout: function() { + var self = germanized.dhl_parcel_locator; + + return self.params.is_checkout; + }, + + afterRefreshCheckout: function() { + var self = germanized.dhl_parcel_locator; + + var params = { + 'security': self.params.parcel_locator_data_nonce, + 'action' : 'woocommerce_gzd_dhl_parcel_locator_refresh_shipping_data' + }; + + $.ajax({ + type: "POST", + url: self.params.ajax_url, + data: params, + success: function( data ) { + // Update shipping method data from session + self.params['methods'] = data.methods; + self.refreshAvailability(); + }, + error: function( data ) { + self.refreshAvailability(); + }, + dataType: 'json' + }); + }, + + refreshAvailability: function() { + var self = germanized.dhl_parcel_locator, + shippingMethod = self.getShippingMethod(), + methodData = self.getShippingMethodData( shippingMethod ); + + if ( ! self.isAvailable() ) { + $( self.wrapper + ' #shipping_address_type' ).val( 'regular' ).trigger( 'change' ); + $( self.wrapper + ' #shipping_address_type_field' ).hide(); + } else { + var $typeField = $( self.wrapper + ' #shipping_address_type' ); + var selected = $typeField.val(); + + if ( self.isCheckout() ) { + $typeField.html( '' ); + + if ( methodData ) { + $.each( methodData.address_type_options, function( name, title ) { + $typeField.append( $( '"),$quantity.data("max-quantity-"+t,i.max_quantity)}),n(".wc-backbone-modal-content article").unblock(),n(document.body).on("change","input#wc-gzd-shipment-add-items-quantity",function(){var t=$select.val(),i=n(this).val();$quantity.data("max-quantity-"+t)&&(t=$quantity.data("max-quantity-"+t)) 0 ) { + + var actionData = self.params.bulk_actions[ action ]; + + $( '.bulk-action-wrapper' ).find( '.bulk-title' ).text( actionData['title'] ); + $( '#posts-filter' ).addClass( 'bulk-action-processing' ); + $( '#posts-filter' ).find( '.bulkactions button' ).prop( 'disabled', true ); + + // Handle bulk action processing + self.handleBulkAction( action, 1, ids, type ); + + return false; + } + }, + + handleBulkAction: function( action, step, ids, type ) { + var self = germanized.admin.shipments_table, + actionData = self.params.bulk_actions[ action ]; + + $.ajax( { + type: 'POST', + url: self.params.ajax_url, + data: { + action : 'woocommerce_gzd_shipments_bulk_action_handle', + bulk_action : action, + step : step, + type : type, + ids : ids, + security : actionData['nonce'] + }, + dataType: 'json', + success: function( response ) { + if ( response.success ) { + + if ( 'done' === response.data.step ) { + $( '.bulk-action-wrapper' ).find( '.woocommerce-shimpents-bulk-progress' ).val( response.data.percentage ); + + window.location = response.data.url; + + setTimeout( function() { + $( '#posts-filter' ).removeClass( 'bulk-action-processing' ); + $( '#posts-filter' ).find( '.bulkactions button' ).prop( 'disabled', false ); + }, 2000 ); + } else { + $( '.bulk-action-wrapper' ).find( '.woocommerce-shimpents-bulk-progress' ).val( response.data.percentage ); + self.handleBulkAction( action, parseInt( response.data.step, 10 ), response.data.ids, response.data.type ); + } + } + } + }).fail( function( response ) { + window.console.log( response ); + } ); + }, + + initTipTip: function() { + $( '.column-actions .wc-gzd-shipment-action-button' ).tipTip( { + 'fadeIn': 50, + 'fadeOut': 50, + 'delay': 200 + }); + }, + + initEnhanced: function() { + try { + $( document.body ) + .on( 'wc-enhanced-select-init', function() { + // Ajax order search boxes + $( ':input.wc-gzd-order-search' ).filter( ':not(.enhanced)' ).each( function() { + var select2_args = { + allowClear: $( this ).data( 'allow_clear' ) ? true : false, + placeholder: $( this ).data( 'placeholder' ), + minimumInputLength: $( this ).data( 'minimum_input_length' ) ? $( this ).data( 'minimum_input_length' ) : '1', + escapeMarkup: function( m ) { + return m; + }, + ajax: { + url: wc_gzd_admin_shipments_table_params.ajax_url, + dataType: 'json', + delay: 1000, + data: function( params ) { + return { + term: params.term, + action: 'woocommerce_gzd_json_search_orders', + security: wc_gzd_admin_shipments_table_params.search_orders_nonce, + exclude: $( this ).data( 'exclude' ) + }; + }, + processResults: function( data ) { + var terms = []; + if ( data ) { + $.each( data, function( id, text ) { + terms.push({ + id: id, + text: text + }); + }); + } + return { + results: terms + }; + }, + cache: true + } + }; + + $( this ).selectWoo( select2_args ).addClass( 'enhanced' ); + }); + + $( ':input.wc-gzd-shipping-provider-search' ).filter( ':not(.enhanced)' ).each( function() { + var select2_args = { + allowClear: $( this ).data( 'allow_clear' ) ? true : false, + placeholder: $( this ).data( 'placeholder' ), + minimumInputLength: $( this ).data( 'minimum_input_length' ) ? $( this ).data( 'minimum_input_length' ) : '1', + escapeMarkup: function( m ) { + return m; + }, + ajax: { + url: wc_gzd_admin_shipments_table_params.ajax_url, + dataType: 'json', + delay: 1000, + data: function( params ) { + return { + term: params.term, + action: 'woocommerce_gzd_json_search_shipping_provider', + security: wc_gzd_admin_shipments_table_params.search_shipping_provider_nonce, + exclude: $( this ).data( 'exclude' ) + }; + }, + processResults: function( data ) { + var terms = []; + if ( data ) { + $.each( data, function( id, text ) { + terms.push({ + id: id, + text: text + }); + }); + } + return { + results: terms + }; + }, + cache: true + } + }; + + $( this ).selectWoo( select2_args ).addClass( 'enhanced' ); + }); + }); + + $( 'html' ).on( 'click', function( event ) { + if ( this === event.target ) { + $( ':input.wc-gzd-order-search' ).filter( '.select2-hidden-accessible' ).selectWoo( 'close' ); + $( ':input.wc-gzd-shipping-provider-search' ).filter( '.select2-hidden-accessible' ).selectWoo( 'close' ); + } + } ); + } catch( err ) { + // If select2 failed (conflict?) log the error but don't stop other scripts breaking. + window.console.log( err ); + } + } + }; + + $( document ).ready( function() { + germanized.admin.shipments_table.init(); + }); + +})( jQuery, window.germanized.admin ); diff --git a/packages/woocommerce-germanized-shipments/assets/js/admin-shipments-table.min.js b/packages/woocommerce-germanized-shipments/assets/js/admin-shipments-table.min.js new file mode 100644 index 000000000..e66290fea --- /dev/null +++ b/packages/woocommerce-germanized-shipments/assets/js/admin-shipments-table.min.js @@ -0,0 +1 @@ +window.germanized=window.germanized||{},window.germanized.admin=window.germanized.admin||{},function(o){window.germanized.admin.shipments_table={params:{},init:function(){var e=germanized.admin.shipments_table;e.params=wc_gzd_admin_shipments_table_params,e.initEnhanced(),o(document).on("click","#doaction, #doaction2",e.onBulkSubmit).on("click",".wc-gzd-shipment-action-button-generate-label",e.onCreateLabel),o(document.body).on("init_tooltips",function(){e.initTipTip()}),e.initTipTip()},onCreateLabel:function(){germanized.admin.shipments_table;var e=o(this).parents("tr").find("th.check-column input").val();return o(this).parents("td").WCBackboneModal({template:"wc-gzd-modal-create-shipment-label-"+e}),!1},onBulkSubmit:function(){var e,t=germanized.admin.shipments_table,n=o(this).parents(".bulkactions").find("select[name^=action]").val(),a=o(this).parents("#posts-filter").find("input.shipment_type").val(),i=[];if(o("#posts-filter").find('input[name="shipment[]"]:checked').each(function(){i.push(o(this).val())}),t.params.bulk_actions.hasOwnProperty(n)&&0 0 ) { + return $shipment.data( 'shipment' ); + } + + return false; + }, + + block: function() { + var self = germanized.admin.shipments; + + self.$wrapper.block({ + message: null, + overlayCSS: { + background: '#fff', + opacity: 0.6 + } + }); + }, + + unblock: function() { + var self = germanized.admin.shipments; + + self.$wrapper.unblock(); + }, + + getData: function( additionalData ) { + var self = germanized.admin.shipments, + data = {}; + + additionalData = additionalData || {}; + + $.each( self.$wrapper.find( ':input[name]' ).serializeArray(), function( index, item ) { + if ( item.name.indexOf( '[]' ) !== -1 ) { + item.name = item.name.replace( '[]', '' ); + data[ item.name ] = $.makeArray( data[ item.name ] ); + data[ item.name ].push( item.value ); + } else { + data[ item.name ] = item.value; + } + }); + + $.extend( data, additionalData ); + + return data; + }, + + doAjax: function( params, cSuccess, cError ) { + var self = germanized.admin.shipments, + url = self.params.ajax_url, + $wrapper = self.$wrapper, + refreshFragments = true; + + $wrapper.find( '.notice-wrapper' ).empty(); + + cSuccess = cSuccess || self.onAjaxSuccess; + cError = cError || self.onAjaxError; + + if ( params.hasOwnProperty( 'refresh_fragments' ) ) { + refreshFragments = params['refresh_fragments']; + } + + if ( ! params.hasOwnProperty( 'security' ) ) { + params['security'] = self.params.edit_shipments_nonce; + } + + if ( ! params.hasOwnProperty( 'order_id' ) ) { + params['order_id'] = self.params.order_id; + } + + params = self.getData( params ); + + $.ajax({ + type: "POST", + url: url, + data: params, + success: function( data ) { + if ( data.success ) { + + active = self.getShipment( self.getActiveShipmentId() ); + current_packaging_id = false; + + if ( active ) { + current_packaging_id = active.getShipment().find( '.shipment-packaging-select' ).val(); + } + + if ( refreshFragments ) { + if ( data.fragments ) { + $.each( data.fragments, function ( key, value ) { + $( key ).replaceWith( value ); + $( key ).unblock(); + } ); + } + } + + cSuccess.apply( $wrapper, [ data ] ); + + if ( data.hasOwnProperty( 'order_needs_new_shipments' ) ) { + self.setNeedsShipments( data.order_needs_new_shipments ); + } + + if ( data.hasOwnProperty( 'order_needs_new_returns' ) ) { + self.setNeedsReturns( data.order_needs_new_returns ); + } + + var shipmentData = data.hasOwnProperty( 'shipments' ) ? data.shipments : {}; + + $.each( self.getShipments(), function( shipmentId, shipment ) { + + if ( shipmentData.hasOwnProperty( shipmentId ) ) { + shipment.setIsEditable( shipmentData[ shipmentId ].is_editable ); + shipment.setNeedsItems( shipmentData[ shipmentId ].needs_items ); + shipment.setWeight( shipmentData[ shipmentId ].weight ); + shipment.setLength( shipmentData[ shipmentId ].length ); + shipment.setWidth( shipmentData[ shipmentId ].width ); + shipment.setHeight( shipmentData[ shipmentId ].height ); + shipment.setTotalWeight( shipmentData[ shipmentId ].total_weight ); + + self.initShipment( shipmentId ); + } + }); + + if ( ( data.hasOwnProperty( 'needs_refresh' ) || data.hasOwnProperty( 'needs_packaging_refresh' ) ) && data.hasOwnProperty( 'shipment_id' ) ) { + self.initShipment( data.shipment_id ); + + if ( data.hasOwnProperty( 'needs_packaging_refresh' ) ) { + active = self.getShipment( self.getActiveShipmentId() ); + + if ( active ) { + // Refresh dimensions in case the packaging has changed + new_packaging_id = active.getShipment().find( '.shipment-packaging-select' ).val(); + + if ( new_packaging_id !== current_packaging_id ) { + self.getShipment( data.shipment_id ).refreshDimensions(); + } + } + } + } + } else { + cError.apply( $wrapper, [ data ] ); + self.unblock(); + + if ( data.hasOwnProperty( 'message' ) ) { + self.addNotice( data.message, 'error' ); + } else if( data.hasOwnProperty( 'messages' ) ) { + $.each( data.messages, function( i, message ) { + self.addNotice( message, 'error' ); + }); + } + } + }, + error: function( data ) { + cError.apply( $wrapper, [ data ] ); + self.unblock(); + }, + dataType: 'json' + }); + }, + + onAjaxError: function( data ) { + + }, + + onAjaxSuccess: function( data ) { + + }, + + onRemoveNotice: function() { + $( this ).parents( '.notice' ).slideUp( 150, function() { + $( this ).remove(); + }); + }, + + addNotice: function( message, noticeType ) { + var self = germanized.admin.shipments; + + self.$wrapper.find( '.notice-wrapper' ).append( '

' + message + '

' ); + }, + + getParams: function() { + var self = germanized.admin.shipments; + + return self.params; + }, + + onRemoveShipment: function() { + var self = germanized.admin.shipments, + $shipment = $( this ).parents( '.order-shipment' ), + id = $shipment.data( 'shipment' ); + + var answer = window.confirm( self.getParams().i18n_remove_shipment_notice ); + + if ( answer ) { + self.removeShipment( id ); + } + + return false; + }, + + removeShipment: function( shipment_id ) { + var self = germanized.admin.shipments; + + var params = { + 'action' : 'woocommerce_gzd_remove_shipment', + 'shipment_id': shipment_id + }; + + self.block(); + self.doAjax( params, self.onRemoveShipmentSuccess, self.onRemoveShipmentError ); + }, + + onRemoveShipmentSuccess: function( data ) { + var self = germanized.admin.shipments, + shipmentIds = Array.isArray( data['shipment_id'] ) ? data['shipment_id'] : [data['shipment_id']]; + + $.each( shipmentIds, function( i, shipmentId ) { + var $shipment = self.$wrapper.find( '#shipment-' + shipmentId ); + + if ( $shipment.length > 0 ) { + if ( $shipment.hasClass( 'active' ) ) { + $shipment.find( '.shipment-content-wrapper' ).slideUp( 300, function() { + $shipment.removeClass( 'active' ); + $shipment.remove(); + + self.initShipments(); + }); + } else { + $shipment.remove(); + } + } + }); + + self.initShipments(); + self.unblock(); + }, + + onRemoveShipmentError: function( data ) { + var self = germanized.admin.shipments; + + self.unblock(); + }, + + onAddShipment: function() { + var self = germanized.admin.shipments; + + self.addShipment(); + + return false; + }, + + addShipment: function() { + var self = germanized.admin.shipments; + + var params = { + 'action': 'woocommerce_gzd_add_shipment' + }; + + self.block(); + self.doAjax( params, self.onAddShipmentSuccess, self.onAddShipmentError ); + }, + + onAddShipmentSuccess: function( data ) { + var self = germanized.admin.shipments; + + if ( self.$wrapper.find( '.order-shipment.active' ).length > 0 ) { + self.$wrapper.find( '.order-shipment.active' ).find( '.shipment-content-wrapper' ).slideUp( 300, function() { + + self.$wrapper.find( '.order-shipment.active' ).removeClass( 'active' ); + self.appendNewShipment( data ); + + self.initShipments(); + + // Init tiptip + self.initTiptip(); + self.unblock(); + }); + } else { + self.appendNewShipment( data ); + self.initShipments(); + + // Init tiptip + self.initTiptip(); + self.unblock(); + } + }, + + appendNewShipment: function( data ) { + var self = germanized.admin.shipments; + + if ( 'simple' === data['new_shipment_type'] && self.$wrapper.find( '.panel-order-return-title' ).length > 0 ) { + self.$wrapper.find( '.panel-order-return-title' ).before( data.new_shipment ); + } else { + self.$wrapper.find( '#order-shipments-list' ).append( data.new_shipment ); + } + }, + + onAddShipmentError: function( data ) { + + }, + + onAddReturn: function() { + + $( this ).WCBackboneModal({ + template: 'wc-gzd-modal-add-shipment-return' + }); + + return false; + }, + + addReturn: function( items ) { + var self = germanized.admin.shipments; + + self.block(); + + var params = { + 'action' : 'woocommerce_gzd_add_return_shipment' + }; + + $.extend( params, items ); + + self.doAjax( params, self.onAddReturnSuccess, self.onAddReturnError ); + }, + + onAddReturnSuccess: function( data ) { + var self = germanized.admin.shipments; + + self.onAddShipmentSuccess( data ); + }, + + onAddReturnError: function( data ) { + var self = germanized.admin.shipments; + + self.onAddShipmentError( data ); + }, + + setNeedsSaving: function( needsSaving ) { + var self = germanized.admin.shipments, + shipmentId = self.getActiveShipmentId(), + $shipment = shipmentId ? self.getShipment( shipmentId ).getShipment() : false; + + if ( typeof needsSaving !== "boolean" ) { + needsSaving = true; + } + + self.needsSaving = needsSaving === true; + + if ( self.needsSaving ) { + self.$wrapper.find( '#order-shipments-save' ).show(); + } else { + self.$wrapper.find( '#order-shipments-save' ).hide(); + } + + if ( $shipment ) { + if ( self.needsSaving ) { + self.disableCreateLabel( $shipment ); + } else { + self.enableCreateLabel( $shipment ); + } + } + + if ( self.needsSaving ) { + self.disableCreateLabel( $shipment ); + } else { + self.enableCreateLabel( $shipment ); + } + + self.hideOrShowFooter(); + + $( document.body ).trigger( 'woocommerce_gzd_shipments_needs_saving', [ self.needsSaving, self.getActiveShipmentId() ] ); + + self.initTiptip(); + }, + + disableCreateLabel: function( $shipment ) { + var self = germanized.admin.shipments, + $button = $shipment.find( '.create-shipment-label' ); + + if ( $button.length > 0 ) { + $button.addClass( 'disabled button-disabled' ); + $button.prop( 'title', self.params.i18n_create_label_disabled ); + } + }, + + enableCreateLabel: function( $shipment ) { + var self = germanized.admin.shipments, + $button = $shipment.find( '.create-shipment-label' ); + + if ( $button.length > 0 ) { + $button.removeClass( 'disabled button-disabled' ); + $button.prop( 'title', self.params.i18n_create_label_enabled ); + } + }, + + setNeedsShipments: function( needsShipments ) { + var self = germanized.admin.shipments; + + if ( typeof needsShipments !== "boolean" ) { + needsShipments = true; + } + + self.needsShipments = needsShipments === true; + + if ( self.needsShipments ) { + self.$wrapper.addClass( 'needs-shipments' ); + self.$wrapper.find( '#order-shipment-add' ).show(); + } else { + self.$wrapper.removeClass( 'needs-shipments' ); + self.$wrapper.find( '#order-shipment-add' ).hide(); + } + + self.hideOrShowFooter(); + }, + + hideOrShowReturnTitle: function() { + var self = germanized.admin.shipments; + + if ( self.$wrapper.find( '.order-shipment.shipment-return' ).length === 0 ) { + self.$wrapper.find( '.panel-order-return-title' ).addClass( 'hide-default' ); + } else { + self.$wrapper.find( '.panel-order-return-title' ).removeClass( 'hide-default' ); + } + }, + + setNeedsReturns: function( needsReturns ) { + var self = germanized.admin.shipments; + + if ( typeof needsReturns !== "boolean" ) { + needsReturns = true; + } + + self.needsReturns = needsReturns === true; + + if ( self.needsReturns ) { + self.$wrapper.addClass( 'needs-returns' ); + self.$wrapper.find( '#order-return-shipment-add' ).show(); + } else { + self.$wrapper.removeClass( 'needs-returns' ); + self.$wrapper.find( '#order-return-shipment-add' ).hide(); + } + + self.hideOrShowFooter(); + }, + + hideOrShowFooter: function() { + var self = germanized.admin.shipments; + + if ( self.needsSaving || self.needsShipments || self.needsReturns ) { + self.$wrapper.find( '.panel-footer' ).slideDown( 300 ); + } else { + self.$wrapper.find( '.panel-footer' ).slideUp( 300 ); + } + }, + + onToggleShipment: function() { + var self = germanized.admin.shipments, + $shipment = $( this ).parents( '.order-shipment:first' ), + isActive = $shipment.hasClass( 'active' ); + + self.closeShipments(); + + if ( ! isActive ) { + $shipment.find( '> .shipment-content-wrapper' ).slideDown( 300, function() { + $shipment.addClass( 'active' ); + }); + } + }, + + closeShipments: function() { + var self = germanized.admin.shipments; + + self.$wrapper.find( '.order-shipment.active .shipment-content-wrapper' ).slideUp( 300, function() { + self.$wrapper.find( '.order-shipment.active' ).removeClass( 'active' ); + }); + }, + + initShipments: function() { + var self = germanized.admin.shipments; + + // Refresh wrapper + self.$wrapper = $( '#panel-order-shipments' ); + + self.$wrapper.find( '.order-shipment' ).each( function() { + var id = $( this ).data( 'shipment' ); + + self.initShipment( id ); + }); + + self.hideOrShowReturnTitle(); + }, + + getShipments: function() { + var self = germanized.admin.shipments; + + return self.shipments; + }, + + getShipment: function( shipment_id ) { + var self = germanized.admin.shipments, + shipments = self.getShipments(); + + if ( shipments.hasOwnProperty( shipment_id ) ) { + return shipments[ shipment_id ]; + } + + return false; + }, + + refresh: function( shipment_id ) { + + }, + + refreshItems: function( shipment_id ) { + + }, + + addItem: function() { + + }, + + initTiptip: function() { + var self = germanized.admin.shipments; + + // Tooltips + $( document.body ).trigger( 'init_tooltips' ); + + self.$wrapper.find( '.woocommerce-help-tip' ).tipTip({ + 'attribute': 'data-tip', + 'fadeIn': 50, + 'fadeOut': 50, + 'delay': 200 + }); + + self.$wrapper.find( '.create-shipment-label' ).tipTip( { + 'fadeIn': 50, + 'fadeOut': 50, + 'delay': 200 + } ); + }, + + backbone: { + + onAddReturnSuccess: function( data ) { + $( '#wc-gzd-return-shipment-items' ).html( data.html ); + $( '.wc-backbone-modal-content article' ).unblock(); + + $( document.body ).on( 'change', 'input.wc-gzd-shipment-add-return-item-quantity', function() { + var $select = $( this ), + quantity = $select.val(); + + if ( $select.attr( 'max' ) ) { + var maxQuantity = $select.attr( 'max' ); + + if ( quantity > maxQuantity ) { + $select.val( maxQuantity ); + } + } + }); + }, + + init: function ( e, target ) { + var self = germanized.admin.shipments; + + if( ( 'wc-gzd-modal-add-shipment-return' ) === target ) { + $( '.wc-backbone-modal-content article' ).block({ + message: null, + overlayCSS: { + background: '#fff', + opacity: 0.6 + } + }); + + self.doAjax( { + 'action' : 'woocommerce_gzd_get_available_return_shipment_items' + }, self.backbone.onAddReturnSuccess ); + + return false; + } + }, + + response: function ( e, target, data ) { + var self = germanized.admin.shipments; + + if( ( 'wc-gzd-modal-add-shipment-return' ) === target ) { + self.addReturn( data ); + } + } + } + }; + + $( document ).ready( function() { + germanized.admin.shipments.init(); + }); + +})( jQuery, window.germanized.admin ); diff --git a/packages/woocommerce-germanized-shipments/assets/js/admin-shipments.min.js b/packages/woocommerce-germanized-shipments/assets/js/admin-shipments.min.js new file mode 100644 index 000000000..625e0f9ee --- /dev/null +++ b/packages/woocommerce-germanized-shipments/assets/js/admin-shipments.min.js @@ -0,0 +1 @@ +window.germanized=window.germanized||{},window.germanized.admin=window.germanized.admin||{},function(d){window.germanized.admin.shipments={params:{},shipments:{},$wrapper:!1,needsSaving:!1,needsShipments:!0,needsReturns:!1,init:function(){var e=germanized.admin.shipments;e.params=wc_gzd_admin_shipments_params,e.$wrapper=d("#panel-order-shipments"),e.needsShipments=e.$wrapper.find("#order-shipment-add").is(":visible"),e.needsReturns=e.$wrapper.find("#order-return-shipment-add").is(":visible"),e.initShipments(),d(document).ajaxComplete(e.onAjaxComplete),d(document).on("click","#order-shipments-list .shipment-header",e.onToggleShipment).on("change","#order-shipments-list :input:visible",e.setNeedsSaving).on("click","#panel-order-shipments #order-shipment-add",e.onAddShipment).on("click","#panel-order-shipments #order-return-shipment-add",e.onAddReturn).on("click","#panel-order-shipments .remove-shipment",e.onRemoveShipment).on("click","#panel-order-shipments button#order-shipments-save",e.onSave).on("click","#panel-order-shipments .notice-dismiss",e.onRemoveNotice),d(document.body).on("wc_backbone_modal_loaded",e.backbone.init).on("wc_backbone_modal_response",e.backbone.response)},onAjaxComplete:function(e,n,i){var t=germanized.admin.shipments;if(null!=n&&i.hasOwnProperty("data")){var n=i.data,i=!1;try{i=JSON.parse('{"'+n.replace(/&/g,'","').replace(/=/g,'":"')+'"}',function(e,n){return""===e?n:decodeURIComponent(n)})}catch(e){i=!1}i&&i.hasOwnProperty("action")&&("woocommerce_save_order_items"!==(n=i.action)&&"woocommerce_remove_order_item"!==n&&"woocommerce_add_order_item"!==n&&"woocommerce_delete_refund"!==n||t.syncItemQuantities())}},syncItemQuantities:function(){var e=germanized.admin.shipments,n=(e.block(),{action:"woocommerce_gzd_validate_shipment_item_quantities",active:e.getActiveShipmentId()});e.doAjax(n,e.onSyncSuccess)},onSyncSuccess:function(e){var n=germanized.admin.shipments;n.unblock(),n.initShipments(),n.initTiptip()},onSave:function(e){var n=germanized.admin.shipments;return e.preventDefault(),n.save(),!1},save:function(){var e=germanized.admin.shipments,n=(e.block(),{action:"woocommerce_gzd_save_shipments",active:e.getActiveShipmentId()});e.doAjax(n,e.onSaveSuccess)},initShipment:function(e){var n=germanized.admin.shipments;n.shipments.hasOwnProperty(e)?n.shipments[e].refreshDom():n.shipments[e]=new d.GermanizedShipment(e)},onSaveSuccess:function(e){var n=germanized.admin.shipments;n.initShipments(),n.setNeedsSaving(!1),n.unblock(),n.initTiptip()},getActiveShipmentId:function(){var e=germanized.admin.shipments.$wrapper.find(".order-shipment.active");return 0

'+e+'

')},getParams:function(){return germanized.admin.shipments.params},onRemoveShipment:function(){var e=germanized.admin.shipments,n=d(this).parents(".order-shipment").data("shipment");return window.confirm(e.getParams().i18n_remove_shipment_notice)&&e.removeShipment(n),!1},removeShipment:function(e){var n=germanized.admin.shipments,e={action:"woocommerce_gzd_remove_shipment",shipment_id:e};n.block(),n.doAjax(e,n.onRemoveShipmentSuccess,n.onRemoveShipmentError)},onRemoveShipmentSuccess:function(e){var t=germanized.admin.shipments,e=Array.isArray(e.shipment_id)?e.shipment_id:[e.shipment_id];d.each(e,function(e,n){var i=t.$wrapper.find("#shipment-"+n);0 .shipment-content-wrapper").slideDown(300,function(){n.addClass("active")})},closeShipments:function(){var e=germanized.admin.shipments;e.$wrapper.find(".order-shipment.active .shipment-content-wrapper").slideUp(300,function(){e.$wrapper.find(".order-shipment.active").removeClass("active")})},initShipments:function(){var n=germanized.admin.shipments;n.$wrapper=d("#panel-order-shipments"),n.$wrapper.find(".order-shipment").each(function(){var e=d(this).data("shipment");n.initShipment(e)}),n.hideOrShowReturnTitle()},getShipments:function(){return germanized.admin.shipments.shipments},getShipment:function(e){var n=germanized.admin.shipments.getShipments();return!!n.hasOwnProperty(e)&&n[e]},refresh:function(e){},refreshItems:function(e){},addItem:function(){},initTiptip:function(){var e=germanized.admin.shipments;d(document.body).trigger("init_tooltips"),e.$wrapper.find(".woocommerce-help-tip").tipTip({attribute:"data-tip",fadeIn:50,fadeOut:50,delay:200}),e.$wrapper.find(".create-shipment-label").tipTip({fadeIn:50,fadeOut:50,delay:200})},backbone:{onAddReturnSuccess:function(e){d("#wc-gzd-return-shipment-items").html(e.html),d(".wc-backbone-modal-content article").unblock(),d(document.body).on("change","input.wc-gzd-shipment-add-return-item-quantity",function(){var e,n=d(this),i=n.val();n.attr("max")&&(e=n.attr("max")) 0 ) { + $( 'select[id$=shipping_provider]' ).trigger( 'change' ); + } + }, + + parseFieldId: function( fieldId ) { + return fieldId.replace( '[', '_' ).replace( ']', '' ); + }, + + onChangeField: function() { + var self = germanized.admin.shipping_provider_method, + $wrapper = $( this ).parents( 'form' ), + fieldId = self.parseFieldId( $( this ).attr( 'id' ) ), + val = $( this ).val(), + currentProvider = self.currentProvider; + + if ( currentProvider && fieldId.toLowerCase().indexOf( '_' + currentProvider + '_' ) >= 0 ) { + // Remove the shipping method name prefix + var fieldIdClean = fieldId.substring( fieldId.lastIndexOf( currentProvider + '_' ), fieldId.length ); + + $wrapper.find( ':input[data-show_if_' + fieldIdClean + ']' ).parents( 'tr' ).hide(); + + if ( $( this ).is( ':checkbox' ) ) { + if ( $( this ).is( ':checked' ) ) { + $wrapper.find( ':input[data-show_if_' + fieldIdClean + ']' ).parents( 'tr' ).show(); + } + } else { + $wrapper.find( ':input[data-show_if_' + fieldIdClean + '*="' + val + '"]' ).parents( 'tr' ).show(); + } + } + }, + + onShippingMethodOpen: function( e, t ) { + if ( 'wc-modal-shipping-method-settings' === t ) { + $wrapper = $( '.wc-modal-shipping-method-settings' ); + + if ( $( 'select[id$=shipping_provider]' ).length > 0 ) { + $wrapper.find( 'select[id$=shipping_provider]' ).trigger( 'change' ); + } + } + }, + + showOrHideAll: function() { + var self = germanized.admin.shipping_provider_method, + $select = $( this ), + $providers = $select.find( 'option' ), + $form = $select.parents( 'form' ); + + self.currentProvider = $select.val(); + + $providers.each( function() { + var $provider = $( this ), + provider_setting_prefix = $provider.val(); + + if ( provider_setting_prefix.length > 0 ) { + $form.find( 'table.form-table' ).each( function() { + if ( $( this ).find( ':input[id*=_' + provider_setting_prefix + '_]' ).length > 0 ) { + self.hideTable( $( this ) ); + } + }); + } + }); + + if ( self.currentProvider.length > 0 ) { + $form.find( 'table.form-table' ).each( function() { + if ( $( this ).find( ':input[id*=_' + self.currentProvider + '_]' ).length > 0 ) { + self.showTable( $( this ) ); + } + }); + + // Trigger show/hide + $form.find( ':input[id*=_' + self.currentProvider + '_]:visible' ).trigger( 'change' ); + } + }, + + hideTable: function( $table ) { + if ( $table.find( 'select[id$=shipping_provider]' ).length > 0 ) { + return false; + } + + $table.prevUntil( 'table.form-table' ).hide(); + $table.hide(); + }, + + showTable: function( $table ) { + $table.prevUntil( 'table.form-table' ).show(); + $table.show(); + } + }; + + $( document ).ready( function() { + germanized.admin.shipping_provider_method.init(); + }); + +})( jQuery, window.germanized.admin ); diff --git a/packages/woocommerce-germanized-shipments/assets/js/admin-shipping-provider-method.min.js b/packages/woocommerce-germanized-shipments/assets/js/admin-shipping-provider-method.min.js new file mode 100644 index 000000000..2a4c359bb --- /dev/null +++ b/packages/woocommerce-germanized-shipments/assets/js/admin-shipping-provider-method.min.js @@ -0,0 +1 @@ +window.germanized=window.germanized||{},window.germanized.admin=window.germanized.admin||{},function(d){window.germanized.admin.shipping_provider_method={params:{},currentProvider:"",init:function(){var i=germanized.admin.shipping_provider_method;i.params=wc_gzd_admin_shipping_provider_method_params,d(document).on("change","select[id$=shipping_provider]",i.showOrHideAll).on("change",":input:visible[id]",i.onChangeField),d(document.body).on("wc_backbone_modal_loaded",i.onShippingMethodOpen),0 0 ) { + return $element.parents( 'tr' ).data( 'shipping-provider' ); + } + + return false; + }, + + onRemoveProvider: function() { + var self = germanized.admin.shipping_providers, + provider = self.getProviderName( $( this ) ); + + if ( provider ) { + var answer = window.confirm( self.getParams().i18n_remove_shipping_provider_notice ); + + if ( answer ) { + self.removeProvider( provider ); + } + } + + return false; + }, + + removeProvider: function( id ) { + var self = germanized.admin.shipping_providers, + params = { + 'action' : 'woocommerce_gzd_remove_shipping_provider', + 'provider': id, + 'security': self.getParams().remove_shipping_provider_nonce + }; + + self.block(); + self.doAjax( params, self.onRemoveProviderSuccess ); + }, + + onRemoveProviderSuccess: function( data ) { + var self = germanized.admin.shipping_providers, + $provider = self.$wrapper.find( 'tr[data-shipping-provider="' + data['provider'] + '"]' ); + + if ( $provider.length > 0 ) { + $provider.remove(); + } + }, + + block: function() { + var self = germanized.admin.shipping_providers; + + self.$wrapper.block({ + message: null, + overlayCSS: { + background: '#fff', + opacity: 0.6 + } + }); + }, + + unblock: function() { + var self = germanized.admin.shipping_providers; + + self.$wrapper.unblock(); + }, + + doAjax: function( params, cSuccess, cError ) { + var self = germanized.admin.shipping_providers, + url = self.params.ajax_url, + $wrapper = self.$wrapper; + + cSuccess = cSuccess || self.onAjaxSuccess; + cError = cError || self.onAjaxError; + + if ( ! params.hasOwnProperty( 'security' ) ) { + params['security'] = self.params.edit_shipping_providers_nonce; + } + + $.ajax({ + type: "POST", + url: url, + data: params, + success: function( data ) { + if ( data.success ) { + cSuccess.apply( $wrapper, [ data ] ); + self.unblock(); + } else { + cError.apply( $wrapper, [ data ] ); + self.unblock(); + } + }, + error: function( data ) { + cError.apply( $wrapper, [ data ] ); + }, + dataType: 'json' + }); + }, + + onAjaxError: function( data ) { + + }, + + onAjaxSuccess: function( data ) { + + }, + + getParams: function() { + var self = germanized.admin.shipping_providers; + + return self.params; + } + }; + + $( document ).ready( function() { + germanized.admin.shipping_providers.init(); + }); + +})( jQuery, window.germanized.admin ); diff --git a/packages/woocommerce-germanized-shipments/assets/js/admin-shipping-providers.min.js b/packages/woocommerce-germanized-shipments/assets/js/admin-shipping-providers.min.js new file mode 100644 index 000000000..241d6e6a8 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/assets/js/admin-shipping-providers.min.js @@ -0,0 +1 @@ +window.germanized=window.germanized||{},window.germanized.admin=window.germanized.admin||{},function(a){window.germanized.admin.shipping_providers={params:{},$wrapper:"",init:function(){var e=germanized.admin.shipping_providers;e.params=wc_gzd_admin_shipping_providers_params,e.$wrapper=a(".wc-gzd-shipping-providers"),a(document).on("click",".wc-gzd-shipping-provider-delete",e.onRemoveProvider).on("change",".wc-gzd-shipping-providers input.wc-gzd-shipping-provider-activated-checkbox",this.onChangeProviderStatus),a(window).on("load",function(){a(document).on("updateMoveButtons","table.wc-gzd-shipping-providers",e.onChangeSort)}),a("table.wc-gzd-shipping-providers tbody").sortable({items:"tr",cursor:"move",axis:"y",handle:"td.sort",scrollSensitivity:40,helper:function(e,i){return i.children().each(function(){a(this).width(a(this).width())}),i.css("left","0"),i},start:function(e,i){i.item.css("background-color","#f6f6f6")},stop:function(e,i){i.item.removeAttr("style"),i.item.trigger("updateMoveButtons")}})},onChangeSort:function(){var e=a(this).find('input[name="provider_order[]"]').map(function(){return a(this).val()}).get(),i=germanized.admin.shipping_providers,e={action:"woocommerce_gzd_sort_shipping_provider",order:e,security:i.getParams().sort_shipping_provider_nonce};i.doAjax(e)},onChangeProviderStatus:function(){var e=germanized.admin.shipping_providers,i=a(this),r=e.getProviderName(i),n=i.parents("td").find(".woocommerce-gzd-input-toggle"),i={action:"woocommerce_gzd_edit_shipping_provider_status",enable:i.is(":checked")?"yes":"no",provider:r};window.onbeforeunload="",n.addClass("woocommerce-input-toggle--loading"),e.doAjax(i,e.onChangeProviderStatusSucess)},onChangeProviderStatusSucess:function(e){var i=germanized.admin.shipping_providers.$wrapper.find('tr[data-shipping-provider="'+e.provider+'"]').find(".woocommerce-gzd-input-toggle");i.removeClass("woocommerce-input-toggle--loading"),i.removeClass("woocommerce-input-toggle--enabled, woocommerce-input-toggle--disabled"),"yes"===e.activated?i.addClass("woocommerce-input-toggle--enabled"):i.addClass("woocommerce-input-toggle--disabled")},getProviderName:function(e){germanized.admin.shipping_providers;return e.data("shipping-provider")?e.data("shipping-provider"):0 + +get_available_items_for_return() as $item_id => $item_data ) : + ?> + + + + + diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-content.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-content.php new file mode 100644 index 000000000..5c7a9b3e5 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-content.php @@ -0,0 +1,261 @@ + + +
+
+ + +
+
+
+

+ + +

+
+
+

+ + + + has_packaging() ? 'disabled="disabled"' : '' ); ?> size="6" class="wc_input_decimal wc-gzd-shipment-dimension has_packaging() ? 'disabled' : '' ); ?>" value="get_length( 'edit' ) ) ); ?>" name="shipment_length[get_id() ); ?>]" id="shipment-length-get_id() ); ?>" placeholder="get_content_length() ) ); ?>" /> + has_packaging() ? 'disabled="disabled"' : '' ); ?> size="6" class="wc_input_decimal wc-gzd-shipment-dimension has_packaging() ? 'disabled' : '' ); ?>" value="get_width( 'edit' ) ) ); ?>" name="shipment_width[get_id() ); ?>]" id="shipment-width-get_id() ); ?>" placeholder="get_content_width() ) ); ?>" /> + has_packaging() ? 'disabled="disabled"' : '' ); ?> size="6" class="wc_input_decimal wc-gzd-shipment-dimension has_packaging() ? 'disabled' : '' ); ?>" value="get_height( 'edit' ) ) ); ?>" name="shipment_height[get_id() ); ?>]" id="shipment-height-get_id() ); ?>" placeholder="get_content_height() ) ); ?>" /> + +

+
+
+ +

+ + + +

+
+ +
+

+ + +

+ + get_available_shipping_methods() ) > 1 ) : ?> +

+ + +

+ + +

+ + +

+ +

+ + +

+ + +
+ +
+
+
+
+ supports_label() && ( ( $label = $shipment->get_label() ) || $shipment->needs_label() ) ) : + include 'label/html-shipment-label.php'; + endif; + ?> +
+
+
+
+ +
+
+
+
+ +
+
+
+
+
+ $column ) : ?> +
+ +
+ +
+
+ +
+ get_items() as $item ) : ?> + + +
+
+ +
+
+ +
+ +
+ +
+ + +
+
+ +
+ + + + +
+
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-item-count.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-item-count.php new file mode 100644 index 000000000..4028a912b --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-item-count.php @@ -0,0 +1,19 @@ + + + + get_shippable_item_count() ) > 0 ) : + $item_count = $shipment->get_item_count(); + ?> + + + diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-item.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-item.php new file mode 100644 index 000000000..8fe650fd2 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-item.php @@ -0,0 +1,69 @@ + + +
+
+ $column ) : ?> + +
+ + + + get_name() ); ?> get_sku() ? '(' . esc_html( $item->get_sku() ) . ')' : '' ); ?> + + + + + + + + + + + + + + + + get_id(), $item, $shipment, $column_name ); + ?> +
+ + +
+ + +
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-list.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-list.php new file mode 100644 index 000000000..44a6178c0 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-list.php @@ -0,0 +1,36 @@ +get_return_shipments(); +?> + +
+ get_simple_shipments() as $shipment ) : + $is_active = ( $active_shipment && $shipment->get_id() === $active_shipment ) ? true : false; + + include 'html-order-shipment.php'; + ?> + + +
+

+ get_return_status() ) ); ?> +
+ + + get_id() === $active_shipment ) ? true : false; + include 'html-order-shipment.php'; + ?> + + +
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-packaging-select.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-packaging-select.php new file mode 100644 index 000000000..25fd7d779 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment-packaging-select.php @@ -0,0 +1,41 @@ +get_available_packaging(); +$available_packaging_ids = array(); + +foreach ( $available_packaging as $packaging ) { + $available_packaging_ids[] = (int) $packaging->get_id(); +} + +$current_packaging_id = $shipment->get_packaging_id(); +$default_packaging_id = \Vendidero\Germanized\Shipments\Package::get_setting( 'default_packaging' ); +$default_packaging = ! empty( $default_packaging_id ) ? wc_gzd_get_packaging( $default_packaging_id ) : false; +$default_exists_in_list = false; +?> + diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment.php new file mode 100644 index 000000000..c5191c14f --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipment.php @@ -0,0 +1,31 @@ + + +
+
+
+

get_type() ) ), esc_html( $shipment->get_shipment_number() ) ); ?>

+ get_status() ) ); ?> +
+ +
+ + +
+
+ +
+ +
+
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipments.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipments.php new file mode 100644 index 000000000..9f04ed95b --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-order-shipments.php @@ -0,0 +1,94 @@ + + +
+
+ +
+

+ get_shipping_status() ) ); ?> +
+ +
+ + + + +
+
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/html-settings-provider-list.php b/packages/woocommerce-germanized-shipments/includes/admin/views/html-settings-provider-list.php new file mode 100644 index 000000000..f5d307883 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/html-settings-provider-list.php @@ -0,0 +1,76 @@ + + + + + + + + + + + + + + $provider ) : ?> + + + + + + + + + +
+
+ + + +
+
+ get_title() ); ?> +
+ + is_manual_integration() ) : ?> + | + + +
+
+

get_description() ); ?>

+
+
+ + is_activated() ? 'yes' : 'no', 'yes' ); ?> + /> +
+
+ get_help_link() ) : ?> + + + + +
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone-error.php b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone-error.php new file mode 100644 index 000000000..600c9684e --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone-error.php @@ -0,0 +1,22 @@ + +
+ +
+ get_error_messages() as $message ) : ?> +
+ +
+ +
+
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone-form.php b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone-form.php new file mode 100644 index 000000000..42a9b6f0c --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone-form.php @@ -0,0 +1,17 @@ + +
+ + + +
diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone.php b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone.php new file mode 100644 index 000000000..b1f35edf3 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label-backbone.php @@ -0,0 +1,34 @@ + + + diff --git a/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label.php b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label.php new file mode 100644 index 000000000..8133d77b9 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/admin/views/label/html-shipment-label.php @@ -0,0 +1,49 @@ + + +
+

get_shipping_provider() ) ) ); ?> has_label() && $shipment->get_tracking_id() ) ? wp_kses_post( Admin::get_shipment_tracking_html( $shipment ) ) : '' ); ?>

+ +
+
+ +
+ get_file() ) : ?> + + + + + +
+ +
+ + + +
+ +
+
+
diff --git a/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-guest-return-shipment-request.php b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-guest-return-shipment-request.php new file mode 100644 index 000000000..55e2ea4a4 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-guest-return-shipment-request.php @@ -0,0 +1,214 @@ +customer_email = true; + $this->id = 'customer_guest_return_shipment_request'; + $this->title = _x( 'Order guest return request', 'shipments', 'woocommerce-germanized' ); + $this->description = _x( 'Order guest return request are sent to the customer after submitting a new return request as a guest.', 'shipments', 'woocommerce-germanized' ); + + $this->template_html = 'emails/customer-guest-return-shipment-request.php'; + $this->template_plain = 'emails/plain/customer-guest-return-shipment-request.php'; + $this->template_base = Package::get_path() . '/templates/'; + $this->helper = function_exists( 'wc_gzd_get_email_helper' ) ? wc_gzd_get_email_helper( $this ) : false; + + $this->placeholders = array( + '{site_title}' => $this->get_blogname(), + '{order_number}' => '', + '{order_date}' => '', + ); + + // Call parent constructor. + parent::__construct(); + } + + /** + * Get email subject. + * + * @since 3.1.0 + * @return string + */ + public function get_default_subject() { + return _x( 'Your return request to your order {order_number}', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Get email heading. + * + * @since 3.1.0 + * @return string + */ + public function get_default_heading() { + return _x( 'Return request to your order: {order_number}', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Switch Woo and Germanized locale + */ + public function setup_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_switch_to_site_locale' ) && apply_filters( 'woocommerce_email_setup_locale', true ) ) { + wc_gzd_switch_to_site_locale(); + } + + parent::setup_locale(); + } + + /** + * Restore Woo and Germanized locale + */ + public function restore_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_restore_locale' ) && apply_filters( 'woocommerce_email_restore_locale', true ) ) { + wc_gzd_restore_locale(); + } + + parent::restore_locale(); + } + + /** + * Trigger. + * + * @param int $order_id order ID. + */ + public function trigger( $order_id ) { + if ( $this->helper ) { + $this->helper->setup_locale(); + } else { + $this->setup_locale(); + } + + if ( $this->object = wc_get_order( $order_id ) ) { + + $this->placeholders['{order_number}'] = $this->object->get_order_number(); + + if ( ( $order_shipment = wc_gzd_get_shipment_order( $this->object ) ) && ( $this->request_url = wc_gzd_get_order_customer_add_return_url( $this->object ) ) ) { + $this->recipient = $this->object->get_billing_email(); + $this->placeholders['{order_date}'] = wc_format_datetime( $this->object->get_date_created() ); + $this->placeholders['{order_number}'] = $this->object->get_order_number(); + } + } + + if ( $this->helper ) { + $this->helper->setup_email_locale(); + } + + if ( $this->is_enabled() && $this->get_recipient() ) { + $this->send( $this->get_recipient(), $this->get_subject(), $this->get_content(), $this->get_headers(), $this->get_attachments() ); + } + + if ( $this->helper ) { + $this->helper->restore_email_locale(); + } + + if ( $this->helper ) { + $this->helper->restore_locale(); + } else { + $this->restore_locale(); + } + } + + /** + * Return content from the additional_content field. + * + * Displayed above the footer. + * + * @since 2.0.4 + * @return string + */ + public function get_additional_content() { + if ( is_callable( 'parent::get_additional_content' ) ) { + return parent::get_additional_content(); + } + + return ''; + } + + /** + * Get content html. + * + * @return string + */ + public function get_content_html() { + return wc_get_template_html( + $this->template_html, + array( + 'order' => $this->object, + 'add_return_request_url' => $this->request_url, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => false, + 'email' => $this, + ) + ); + } + + /** + * Get content plain. + * + * @return string + */ + public function get_content_plain() { + return wc_get_template_html( + $this->template_plain, + array( + 'order' => $this->object, + 'add_return_request_url' => $this->request_url, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => true, + 'email' => $this, + ) + ); + } + + /** + * Default content to show below main email content. + * + * @since 1.0.1 + * @return string + */ + public function get_default_additional_content() { + return ''; + } + } + +endif; + +return new WC_GZD_Email_Customer_Guest_Return_Shipment_Request(); diff --git a/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-return-shipment-delivered.php b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-return-shipment-delivered.php new file mode 100644 index 000000000..9b319386d --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-return-shipment-delivered.php @@ -0,0 +1,234 @@ +customer_email = true; + $this->id = 'customer_return_shipment_delivered'; + $this->title = _x( 'Order return delivered', 'shipments', 'woocommerce-germanized' ); + $this->description = _x( 'Order return notifications are sent to the customer after a return shipment has been returned (delivered) successfully.', 'shipments', 'woocommerce-germanized' ); + + $this->template_html = 'emails/customer-return-shipment-delivered.php'; + $this->template_plain = 'emails/plain/customer-return-shipment-delivered.php'; + $this->template_base = Package::get_path() . '/templates/'; + $this->helper = function_exists( 'wc_gzd_get_email_helper' ) ? wc_gzd_get_email_helper( $this ) : false; + + $this->placeholders = array( + '{site_title}' => $this->get_blogname(), + '{shipment_number}' => '', + '{order_number}' => '', + '{order_date}' => '', + '{date_sent}' => '', + ); + + // Triggers for this email. + add_action( 'woocommerce_gzd_return_shipment_status_processing_to_delivered_notification', array( $this, 'trigger' ), 10 ); + add_action( 'woocommerce_gzd_return_shipment_status_shipped_to_delivered_notification', array( $this, 'trigger' ), 10 ); + + // Call parent constructor. + parent::__construct(); + } + + /** + * Get email subject. + * + * @since 3.1.0 + * @return string + */ + public function get_default_subject() { + return _x( 'Return to your order {order_number} has been received', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Get email heading. + * + * @since 3.1.0 + * @return string + */ + public function get_default_heading() { + return _x( 'Received return to your order: {order_number}', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Switch Woo and Germanized locale + */ + public function setup_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_switch_to_site_locale' ) && apply_filters( 'woocommerce_email_setup_locale', true ) ) { + wc_gzd_switch_to_site_locale(); + } + + parent::setup_locale(); + } + + /** + * Restore Woo and Germanized locale + */ + public function restore_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_restore_locale' ) && apply_filters( 'woocommerce_email_restore_locale', true ) ) { + wc_gzd_restore_locale(); + } + + parent::restore_locale(); + } + + /** + * Trigger. + * + * @param int $shipment_id Shipment ID. + * @param bool $is_confirmation + */ + public function trigger( $shipment_id ) { + if ( $this->helper ) { + $this->helper->setup_locale(); + } else { + $this->setup_locale(); + } + + if ( $this->shipment = wc_gzd_get_shipment( $shipment_id ) ) { + + if ( 'return' !== $this->shipment->get_type() ) { + return; + } + + $this->placeholders['{shipment_number}'] = $this->shipment->get_shipment_number(); + + if ( $order_shipment = wc_gzd_get_shipment_order( $this->shipment->get_order() ) ) { + + $this->object = $this->shipment->get_order(); + $this->recipient = $order_shipment->get_order()->get_billing_email(); + $this->placeholders['{order_date}'] = wc_format_datetime( $order_shipment->get_order()->get_date_created() ); + $this->placeholders['{order_number}'] = $order_shipment->get_order()->get_order_number(); + + if ( $this->shipment->get_date_sent() ) { + $this->placeholders['{date_sent}'] = wc_format_datetime( $this->shipment->get_date_sent() ); + } + } + } + + if ( $this->helper ) { + $this->helper->setup_email_locale(); + } + + if ( $this->is_enabled() && $this->get_recipient() ) { + $this->send( $this->get_recipient(), $this->get_subject(), $this->get_content(), $this->get_headers(), $this->get_attachments() ); + } + + if ( $this->helper ) { + $this->helper->restore_email_locale(); + } + + if ( $this->helper ) { + $this->helper->restore_locale(); + } else { + $this->restore_locale(); + } + } + + /** + * Return content from the additional_content field. + * + * Displayed above the footer. + * + * @since 2.0.4 + * @return string + */ + public function get_additional_content() { + if ( is_callable( 'parent::get_additional_content' ) ) { + return parent::get_additional_content(); + } + + return ''; + } + + /** + * Get content html. + * + * @return string + */ + public function get_content_html() { + return wc_get_template_html( + $this->template_html, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => false, + 'email' => $this, + ) + ); + } + + /** + * Get content plain. + * + * @return string + */ + public function get_content_plain() { + return wc_get_template_html( + $this->template_plain, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => true, + 'email' => $this, + ) + ); + } + + /** + * Default content to show below main email content. + * + * @since 1.0.1 + * @return string + */ + public function get_default_additional_content() { + return ''; + } + } + +endif; + +return new WC_GZD_Email_Customer_Return_Shipment_Delivered(); diff --git a/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-return-shipment.php b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-return-shipment.php new file mode 100644 index 000000000..3687a3ec4 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-return-shipment.php @@ -0,0 +1,267 @@ +customer_email = true; + $this->id = 'customer_return_shipment'; + $this->title = _x( 'Order return', 'shipments', 'woocommerce-germanized' ); + $this->description = _x( 'Order return notifications are sent to the customer after a return shipment was marked as processing.', 'shipments', 'woocommerce-germanized' ); + + $this->template_html = 'emails/customer-return-shipment.php'; + $this->template_plain = 'emails/plain/customer-return-shipment.php'; + $this->template_base = Package::get_path() . '/templates/'; + $this->helper = function_exists( 'wc_gzd_get_email_helper' ) ? wc_gzd_get_email_helper( $this ) : false; + + $this->placeholders = array( + '{site_title}' => $this->get_blogname(), + '{shipment_number}' => '', + '{order_number}' => '', + '{order_date}' => '', + '{date_sent}' => '', + ); + + // Triggers for this email. + add_action( 'woocommerce_gzd_return_shipment_status_draft_to_processing_notification', array( $this, 'trigger' ), 10 ); + add_action( 'woocommerce_gzd_return_shipment_status_requested_to_processing_notification', array( $this, 'trigger' ), 10 ); + + // Call parent constructor. + parent::__construct(); + } + + /** + * Get email subject. + * + * @since 3.1.0 + * @return string + */ + public function get_default_subject() { + return _x( 'Return to your order {order_number}', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Get email heading. + * + * @since 3.1.0 + * @return string + */ + public function get_default_heading() { + return _x( 'Return to your order: {order_number}', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Switch Woo and Germanized locale + */ + public function setup_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_switch_to_site_locale' ) && apply_filters( 'woocommerce_email_setup_locale', true ) ) { + wc_gzd_switch_to_site_locale(); + } + + parent::setup_locale(); + } + + /** + * Restore Woo and Germanized locale + */ + public function restore_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_restore_locale' ) && apply_filters( 'woocommerce_email_restore_locale', true ) ) { + wc_gzd_restore_locale(); + } + + parent::restore_locale(); + } + + /** + * Trigger. + * + * @param int $shipment_id Shipment ID. + * @param bool $is_confirmation + */ + public function trigger( $shipment_id, $is_confirmation = false ) { + if ( $this->helper ) { + $this->helper->setup_locale(); + } else { + $this->setup_locale(); + } + + $this->is_confirmation = $is_confirmation; + + if ( $this->shipment = wc_gzd_get_shipment( $shipment_id ) ) { + + if ( 'return' !== $this->shipment->get_type() ) { + return; + } + + // Check if this is a customer request. + if ( $this->shipment->is_customer_requested() ) { + $this->is_confirmation = true; + } + + $this->placeholders['{shipment_number}'] = $this->shipment->get_shipment_number(); + + if ( $order_shipment = wc_gzd_get_shipment_order( $this->shipment->get_order() ) ) { + + $this->object = $this->shipment->get_order(); + $this->recipient = $order_shipment->get_order()->get_billing_email(); + $this->placeholders['{order_date}'] = wc_format_datetime( $order_shipment->get_order()->get_date_created() ); + $this->placeholders['{order_number}'] = $order_shipment->get_order()->get_order_number(); + + if ( $this->shipment->get_date_sent() ) { + $this->placeholders['{date_sent}'] = wc_format_datetime( $this->shipment->get_date_sent() ); + } + } + } + + $this->id = 'customer_return_shipment'; + + if ( $this->helper ) { + $this->helper->setup_email_locale(); + } + + if ( $this->is_enabled() && $this->get_recipient() ) { + $this->send( $this->get_recipient(), $this->get_subject(), $this->get_content(), $this->get_headers(), $this->get_attachments() ); + } + + if ( $this->helper ) { + $this->helper->restore_email_locale(); + } + + if ( $this->helper ) { + $this->helper->restore_locale(); + } else { + $this->restore_locale(); + } + } + + /** + * Return content from the additional_content field. + * + * Displayed above the footer. + * + * @since 2.0.4 + * @return string + */ + public function get_additional_content() { + if ( is_callable( 'parent::get_additional_content' ) ) { + return parent::get_additional_content(); + } + + return ''; + } + + /** + * Get content html. + * + * @return string + */ + public function get_content_html() { + return wc_get_template_html( + $this->template_html, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'is_confirmation' => $this->is_confirmation, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => false, + 'email' => $this, + ) + ); + } + + /** + * Get content plain. + * + * @return string + */ + public function get_content_plain() { + return wc_get_template_html( + $this->template_plain, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'is_confirmation' => $this->is_confirmation, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => true, + 'email' => $this, + ) + ); + } + + public function get_attachments() { + $attachments = array(); + + if ( $this->shipment->has_label() ) { + $label = $this->shipment->get_label(); + + if ( $file = $label->get_file() ) { + $attachments[] = $file; + } + } + + return apply_filters( 'woocommerce_email_attachments', $attachments, $this->id, $this->object, $this ); + } + + /** + * Default content to show below main email content. + * + * @since 1.0.1 + * @return string + */ + public function get_default_additional_content() { + return ''; + } + } + +endif; + +return new WC_GZD_Email_Customer_Return_Shipment(); diff --git a/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-shipment.php b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-shipment.php new file mode 100644 index 000000000..2fffd6cd0 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-customer-shipment.php @@ -0,0 +1,415 @@ +customer_email = true; + $this->id = 'customer_shipment'; + $this->title = _x( 'Order shipped', 'shipments', 'woocommerce-germanized' ); + $this->description = _x( 'Shipment notifications are sent to the customer when a shipment gets shipped.', 'shipments', 'woocommerce-germanized' ); + + $this->template_html = 'emails/customer-shipment.php'; + $this->template_plain = 'emails/plain/customer-shipment.php'; + $this->template_base = Package::get_path() . '/templates/'; + $this->helper = function_exists( 'wc_gzd_get_email_helper' ) ? wc_gzd_get_email_helper( $this ) : false; + + $this->placeholders = array( + '{site_title}' => $this->get_blogname(), + '{shipment_number}' => '', + '{order_number}' => '', + '{order_date}' => '', + '{date_sent}' => '', + '{current_shipment_num}' => '', + '{total_shipments}' => '', + ); + + // Triggers for this email. + if ( 'yes' === Package::get_setting( 'notify_enable' ) ) { + add_action( 'woocommerce_gzd_shipment_status_draft_to_shipped_notification', array( $this, 'trigger' ), 10 ); + add_action( 'woocommerce_gzd_shipment_status_processing_to_shipped_notification', array( $this, 'trigger' ), 10 ); + } + + // Call parent constructor. + parent::__construct(); + } + + /** + * Get email subject. + * + * @param bool $partial Whether it is a partial refund or a full refund. + * @since 3.1.0 + * @return string + */ + public function get_default_subject( $partial = false ) { + if ( $partial ) { + return _x( 'Your {site_title} order #{order_number} has been partially shipped ({current_shipment_num}/{total_shipments})', 'shipments', 'woocommerce-germanized' ); + } else { + return _x( 'Your {site_title} order #{order_number} has been shipped ({current_shipment_num}/{total_shipments})', 'shipments', 'woocommerce-germanized' ); + } + } + + /** + * Get email heading. + * + * @param bool $partial Whether it is a partial refund or a full refund. + * @since 3.1.0 + * @return string + */ + public function get_default_heading( $partial = false ) { + if ( $partial ) { + return _x( 'Partial shipment to your order: {order_number}', 'shipments', 'woocommerce-germanized' ); + } else { + return _x( 'Shipment to your order: {order_number}', 'shipments', 'woocommerce-germanized' ); + } + } + + /** + * Get email subject. + * + * @return string + */ + public function get_subject() { + if ( $this->partial_shipment ) { + $subject = $this->get_option( 'subject_partial', $this->get_default_subject( true ) ); + } else { + $subject = $this->get_option( 'subject_full', $this->get_default_subject() ); + } + + /** + * Filter to adjust the email subject for a shipped Shipment. + * + * @param string $subject The subject. + * @param WC_GZD_Email_Customer_Shipment $email The email instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_email_subject_customer_shipment', $this->format_string( $subject ), $this->object, $this ); + } + + /** + * Get email heading. + * + * @return string + */ + public function get_heading() { + if ( $this->partial_shipment ) { + $heading = $this->get_option( 'heading_partial', $this->get_default_heading( true ) ); + } else { + $heading = $this->get_option( 'heading_full', $this->get_default_heading() ); + } + + /** + * Filter to adjust the email heading for a shipped Shipment. + * + * @param string $heading The heading. + * @param WC_GZD_Email_Customer_Shipment $email The email instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_email_heading_customer_shipment', $this->format_string( $heading ), $this->object, $this ); + } + + /** + * Switch Woo and Germanized locale + */ + public function setup_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_switch_to_site_locale' ) && apply_filters( 'woocommerce_email_setup_locale', true ) ) { + wc_gzd_switch_to_site_locale(); + } + + parent::setup_locale(); + } + + /** + * Restore Woo and Germanized locale + */ + public function restore_locale() { + + if ( $this->is_customer_email() && function_exists( 'wc_gzd_restore_locale' ) && apply_filters( 'woocommerce_email_restore_locale', true ) ) { + wc_gzd_restore_locale(); + } + + parent::restore_locale(); + } + + public function get_shipped_position_number( $shipment_id ) { + $shipped_count = 1; + + if ( ! $this->object || ( ! $order = wc_gzd_get_shipment_order( $this->object ) ) ) { + return $shipped_count; + } + + if ( is_a( $shipment_id, '\Vendidero\Germanized\Shipments\Shipment' ) ) { + $shipment_id = $shipment_id->get_id(); + } + + if ( ! is_numeric( $shipment_id ) ) { + return $shipped_count; + } + + foreach ( $order->get_simple_shipments() as $key => $shipment ) { + if ( $shipment->is_shipped() ) { + if ( (int) $shipment->get_id() !== (int) $shipment_id ) { + $shipped_count++; + } + } + } + + return $shipped_count; + } + + /** + * Trigger. + * + * @param int $shipment_id Shipment ID. + */ + public function trigger( $shipment_id ) { + if ( $this->helper ) { + $this->helper->setup_locale(); + } else { + $this->setup_locale(); + } + + $this->partial_shipment = false; + + if ( $this->shipment = wc_gzd_get_shipment( $shipment_id ) ) { + + $this->placeholders['{shipment_number}'] = $this->shipment->get_shipment_number(); + + if ( $order_shipment = wc_gzd_get_shipment_order( $this->shipment->get_order() ) ) { + $this->object = $this->shipment->get_order(); + $this->total_shipments = count( $order_shipment->get_simple_shipments() ); + $this->cur_position = $this->get_shipped_position_number( $shipment_id ); + + if ( $order_shipment->needs_shipping() || $this->total_shipments > 1 ) { + if ( $order_shipment->needs_shipping() || ( $this->cur_position < $this->total_shipments ) ) { + $this->partial_shipment = true; + } + } + + $this->recipient = $order_shipment->get_order()->get_billing_email(); + $this->placeholders['{order_date}'] = wc_format_datetime( $order_shipment->get_order()->get_date_created() ); + $this->placeholders['{order_number}'] = $order_shipment->get_order()->get_order_number(); + $this->placeholders['{total_shipments}'] = $this->total_shipments; + $this->placeholders['{current_shipment_num}'] = $this->cur_position; + + if ( $this->shipment->get_date_sent() ) { + $this->placeholders['{date_sent}'] = wc_format_datetime( $this->shipment->get_date_sent() ); + } + } + } + + $this->id = $this->partial_shipment ? 'customer_partial_shipment' : 'customer_shipment'; + + if ( $this->helper ) { + $this->helper->setup_email_locale(); + } + + if ( $this->is_enabled() && $this->get_recipient() ) { + $this->send( $this->get_recipient(), $this->get_subject(), $this->get_content(), $this->get_headers(), $this->get_attachments() ); + } + + if ( $this->helper ) { + $this->helper->restore_email_locale(); + } + + if ( $this->helper ) { + $this->helper->restore_locale(); + } else { + $this->restore_locale(); + } + } + + /** + * Return content from the additional_content field. + * + * Displayed above the footer. + * + * @since 2.0.4 + * @return string + */ + public function get_additional_content() { + if ( is_callable( 'parent::get_additional_content' ) ) { + return parent::get_additional_content(); + } + + return ''; + } + + /** + * Get content html. + * + * @return string + */ + public function get_content_html() { + return wc_get_template_html( + $this->template_html, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'partial_shipment' => $this->partial_shipment, + 'cur_position' => $this->cur_position, + 'total_shipments' => $this->total_shipments, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => false, + 'email' => $this, + ) + ); + } + + /** + * Get content plain. + * + * @return string + */ + public function get_content_plain() { + return wc_get_template_html( + $this->template_plain, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'partial_shipment' => $this->partial_shipment, + 'cur_position' => $this->cur_position, + 'total_shipments' => $this->total_shipments, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => true, + 'email' => $this, + ) + ); + } + + /** + * Default content to show below main email content. + * + * @since 1.0.1 + * @return string + */ + public function get_default_additional_content() { + return ''; + } + + /** + * Initialise settings form fields. + */ + public function init_form_fields() { + /* translators: %s: list of placeholders */ + $placeholder_text = sprintf( _x( 'Available placeholders: %s', 'shipments', 'woocommerce-germanized' ), '' . esc_html( implode( ', ', array_keys( $this->placeholders ) ) ) . '' ); + + $this->form_fields = array( + 'enabled' => array( + 'title' => _x( 'Enable/Disable', 'shipments', 'woocommerce-germanized' ), + 'type' => 'checkbox', + 'label' => _x( 'Enable this email notification', 'shipments', 'woocommerce-germanized' ), + 'default' => 'yes', + ), + 'subject_full' => array( + 'title' => _x( 'Full shipment subject', 'shipments', 'woocommerce-germanized' ), + 'type' => 'text', + 'desc_tip' => true, + 'description' => $placeholder_text, + 'placeholder' => $this->get_default_subject(), + 'default' => '', + ), + 'subject_partial' => array( + 'title' => _x( 'Partial shipment subject', 'shipments', 'woocommerce-germanized' ), + 'type' => 'text', + 'desc_tip' => true, + 'description' => $placeholder_text, + 'placeholder' => $this->get_default_subject( true ), + 'default' => '', + ), + 'heading_full' => array( + 'title' => _x( 'Full shipment email heading', 'shipments', 'woocommerce-germanized' ), + 'type' => 'text', + 'desc_tip' => true, + 'description' => $placeholder_text, + 'placeholder' => $this->get_default_heading(), + 'default' => '', + ), + 'heading_partial' => array( + 'title' => _x( 'Partial shipment email heading', 'shipments', 'woocommerce-germanized' ), + 'type' => 'text', + 'desc_tip' => true, + 'description' => $placeholder_text, + 'placeholder' => $this->get_default_heading( true ), + 'default' => '', + ), + 'additional_content' => array( + 'title' => _x( 'Additional content', 'shipments', 'woocommerce-germanized' ), + 'description' => _x( 'Text to appear below the main email content.', 'shipments', 'woocommerce-germanized' ) . ' ' . $placeholder_text, + 'css' => 'width:400px; height: 75px;', + 'placeholder' => _x( 'N/A', 'shipments', 'woocommerce-germanized' ), + 'type' => 'textarea', + 'default' => $this->get_default_additional_content(), + 'desc_tip' => true, + ), + 'email_type' => array( + 'title' => _x( 'Email type', 'shipments', 'woocommerce-germanized' ), + 'type' => 'select', + 'description' => _x( 'Choose which format of email to send.', 'shipments', 'woocommerce-germanized' ), + 'default' => 'html', + 'class' => 'email_type wc-enhanced-select', + 'options' => $this->get_email_type_options(), + 'desc_tip' => true, + ), + ); + } + } + +endif; + +return new WC_GZD_Email_Customer_Shipment(); diff --git a/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-new-return-shipment-request.php b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-new-return-shipment-request.php new file mode 100644 index 000000000..447beca08 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/emails/class-wc-gzd-email-new-return-shipment-request.php @@ -0,0 +1,221 @@ +id = 'new_return_shipment_request'; + $this->title = _x( 'New order return request', 'shipments', 'woocommerce-germanized' ); + $this->description = _x( 'New order return request emails are sent to chosen recipient(s) when a new return is requested.', 'shipments', 'woocommerce-germanized' ); + + $this->template_html = 'emails/admin-new-return-shipment-request.php'; + $this->template_plain = 'emails/plain/admin-new-return-shipment-request.php'; + $this->template_base = Package::get_path() . '/templates/'; + + $this->placeholders = array( + '{site_title}' => $this->get_blogname(), + '{shipment_number}' => '', + '{order_number}' => '', + '{order_date}' => '', + ); + + // Triggers for this email. + add_action( 'woocommerce_gzd_new_customer_return_shipment_request', array( $this, 'trigger' ), 10 ); + + // Call parent constructor. + parent::__construct(); + + // Other settings. + $this->recipient = $this->get_option( 'recipient', get_option( 'admin_email' ) ); + } + + /** + * Get email subject. + * + * @since 3.1.0 + * @return string + */ + public function get_default_subject() { + return _x( '[{site_title}]: New return request to #{order_number}', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Get email heading. + * + * @since 3.1.0 + * @return string + */ + public function get_default_heading() { + return _x( 'New return request to: #{order_number}', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Trigger. + * + * @param int|ReturnShipment $shipment_id Shipment ID. + */ + public function trigger( $shipment_id ) { + $this->setup_locale(); + + if ( $this->shipment = wc_gzd_get_shipment( $shipment_id ) ) { + + if ( 'return' !== $this->shipment->get_type() ) { + return; + } + + $this->placeholders['{shipment_number}'] = $this->shipment->get_shipment_number(); + + if ( $order_shipment = wc_gzd_get_shipment_order( $this->shipment->get_order() ) ) { + $this->object = $this->shipment->get_order(); + $this->placeholders['{order_date}'] = wc_format_datetime( $order_shipment->get_order()->get_date_created() ); + $this->placeholders['{order_number}'] = $order_shipment->get_order()->get_order_number(); + } + } + + if ( $this->is_enabled() && $this->get_recipient() ) { + $this->send( $this->get_recipient(), $this->get_subject(), $this->get_content(), $this->get_headers(), $this->get_attachments() ); + } + + $this->restore_locale(); + } + + /** + * Return content from the additional_content field. + * + * Displayed above the footer. + * + * @since 2.0.4 + * @return string + */ + public function get_additional_content() { + if ( is_callable( 'parent::get_additional_content' ) ) { + return parent::get_additional_content(); + } + + return ''; + } + + /** + * Get content html. + * + * @return string + */ + public function get_content_html() { + return wc_get_template_html( + $this->template_html, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => true, + 'plain_text' => false, + 'email' => $this, + ) + ); + } + + /** + * Get content plain. + * + * @return string + */ + public function get_content_plain() { + return wc_get_template_html( + $this->template_plain, + array( + 'shipment' => $this->shipment, + 'order' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => true, + 'plain_text' => true, + 'email' => $this, + ) + ); + } + + public function get_attachments() { + $attachments = array(); + + if ( $this->shipment->has_label() ) { + $label = $this->shipment->get_label(); + + if ( $file = $label->get_file() ) { + $attachments[] = $file; + } + } + + return apply_filters( 'woocommerce_email_attachments', $attachments, $this->id, $this->object, $this ); + } + + /** + * Default content to show below main email content. + * + * @since 1.0.1 + * @return string + */ + public function get_default_additional_content() { + return ''; + } + + /** + * Initialise settings form fields. + */ + public function init_form_fields() { + parent::init_form_fields(); + + $this->form_fields = array_merge( + $this->form_fields, + array( + 'recipient' => array( + 'title' => _x( 'Recipient(s)', 'shipments', 'woocommerce-germanized' ), + 'type' => 'text', + /* translators: %s: WP admin email */ + 'description' => sprintf( _x( 'Enter recipients (comma separated) for this email. Defaults to %s.', 'shipments', 'woocommerce-germanized' ), '' . esc_attr( get_option( 'admin_email' ) ) . '' ), + 'placeholder' => '', + 'default' => '', + 'desc_tip' => true, + ), + ) + ); + } + } + +endif; + +return new WC_GZD_Email_New_Return_Shipment_Request(); diff --git a/packages/woocommerce-germanized-shipments/includes/wc-gzd-label-functions.php b/packages/woocommerce-germanized-shipments/includes/wc-gzd-label-functions.php new file mode 100644 index 000000000..1d9785021 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/wc-gzd-label-functions.php @@ -0,0 +1,152 @@ +get_labels(); +} + +function wc_gzd_get_label_type_by_shipment( $shipment ) { + $type = is_a( $shipment, '\Vendidero\Germanized\Shipments\Shipment' ) ? $shipment->get_type() : $shipment; + + return apply_filters( 'woocommerce_gzd_shipment_label_type', $type, $shipment ); +} + +function wc_gzd_get_shipment_label_types() { + return apply_filters( + 'woocommerce_gzd_shipment_label_types', + array( + 'simple', + 'return', + ) + ); +} + +function wc_gzd_get_label_by_shipment( $the_shipment, $type = '' ) { + $shipment_id = \Vendidero\Germanized\Shipments\ShipmentFactory::get_shipment_id( $the_shipment ); + $label = false; + + if ( $shipment_id ) { + $args = array( + 'shipment_id' => $shipment_id, + 'limit' => 1, + ); + + if ( ! empty( $type ) ) { + $args['type'] = $type; + } + + $labels = wc_gzd_get_shipment_labels( $args ); + + if ( ! empty( $labels ) ) { + $label = $labels[0]; + } + } + + return apply_filters( 'woocommerce_gzd_shipment_label_for_shipment', $label, $the_shipment ); +} + +/** + * @param false $the_label + * @param string $shipping_provider + * @param string $type + * + * @return \Vendidero\Germanized\Shipments\Interfaces\ShipmentLabel|boolean + */ +function wc_gzd_get_shipment_label( $the_label = false, $shipping_provider = '', $type = 'simple' ) { + return apply_filters( 'woocommerce_gzd_shipment_label', \Vendidero\Germanized\Shipments\Labels\Factory::get_label( $the_label, $shipping_provider, $type ), $the_label, $shipping_provider, $type ); +} + +/** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * @param bool $net_weight + * @param string $unit + * + * @return float + */ +function wc_gzd_get_shipment_label_weight( $shipment, $net_weight = false, $unit = 'kg' ) { + $shipment_weight = $shipment->get_total_weight(); + $shipment_content_weight = $shipment->get_weight(); + $shipment_packaging_weight = $shipment->get_packaging_weight(); + + if ( ! empty( $shipment_weight ) ) { + $shipment_weight = wc_get_weight( $shipment_weight, $unit, $shipment->get_weight_unit() ); + } + + if ( ! empty( $shipment_content_weight ) ) { + $shipment_content_weight = wc_get_weight( $shipment_content_weight, $unit, $shipment->get_weight_unit() ); + } + + if ( ! empty( $shipment_packaging_weight ) ) { + $shipment_packaging_weight = wc_get_weight( $shipment_packaging_weight, $unit, $shipment->get_weight_unit() ); + } + + /** + * The net weight does not include packaging weight. + */ + if ( $net_weight ) { + $shipment_packaging_weight = 0; + $shipment_weight = $shipment_content_weight; + } + + if ( $provider = $shipment->get_shipping_provider_instance() ) { + $min_weight = wc_get_weight( $provider->get_shipment_setting( $shipment, 'label_minimum_shipment_weight' ), $unit, 'kg' ); + $default_weight = wc_get_weight( $provider->get_shipment_setting( $shipment, 'label_default_shipment_weight' ), $unit, 'kg' ); + + if ( empty( $shipment_content_weight ) ) { + $shipment_weight = $default_weight; + + if ( ! $net_weight ) { + $shipment_weight += $shipment_packaging_weight; + } + } + + if ( $shipment_weight < $min_weight ) { + $shipment_weight = $min_weight; + } + } + + $shipment_weight = wc_format_decimal( $shipment_weight, 2 ); + + return apply_filters( 'woocommerce_gzd_shipment_label_weight', $shipment_weight, $shipment, $unit ); +} + +/** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * @param string $dimension + * @param string $unit + */ +function wc_gzd_get_shipment_label_dimensions( $shipment, $unit = 'cm' ) { + $dimensions = array( + 'length' => 0, + 'width' => 0, + 'height' => 0, + ); + + if ( $shipment->has_dimensions() ) { + $dimensions = $shipment->get_package_dimensions(); + + foreach ( $dimensions as $key => $data ) { + $dimensions[ $key ] = wc_get_dimension( $data, $unit, $shipment->get_dimension_unit() ); + } + } + + return apply_filters( 'woocommerce_gzd_shipment_label_dimensions', $dimensions, $shipment, $unit ); +} diff --git a/packages/woocommerce-germanized-shipments/includes/wc-gzd-packaging-functions.php b/packages/woocommerce-germanized-shipments/includes/wc-gzd-packaging-functions.php new file mode 100644 index 000000000..6d949f7d2 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/wc-gzd-packaging-functions.php @@ -0,0 +1,60 @@ + _x( 'Cardboard', 'shipments', 'woocommerce-germanized' ), + 'letter' => _x( 'Letter', 'shipments', 'woocommerce-germanized' ), + ); + + return apply_filters( 'woocommerce_gzd_packaging_types', $types ); +} + +/** + * @return \Vendidero\Germanized\Shipments\Packaging[] $packaging_list + */ +function wc_gzd_get_packaging_list() { + $data_store = \WC_Data_Store::load( 'packaging' ); + $list = $data_store->get_packaging_list(); + + return $list; +} + +function wc_gzd_get_packaging_weight_unit() { + return apply_filters( 'woocommerce_gzd_packaging_weight_unit', 'kg' ); +} + +function wc_gzd_get_packaging_dimension_unit() { + return apply_filters( 'woocommerce_gzd_packaging_dimension_unit', 'cm' ); +} + +function wc_gzd_get_packaging_select() { + $list = wc_gzd_get_packaging_list(); + $select = array( + '' => _x( 'None', 'shipments-packaging', 'woocommerce-germanized' ), + ); + + foreach ( $list as $packaging ) { + $select[ $packaging->get_id() ] = $packaging->get_title(); + } + + return $select; +} diff --git a/packages/woocommerce-germanized-shipments/includes/wc-gzd-shipment-functions.php b/packages/woocommerce-germanized-shipments/includes/wc-gzd-shipment-functions.php new file mode 100644 index 000000000..9dde51ea1 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/wc-gzd-shipment-functions.php @@ -0,0 +1,1527 @@ +countries ? WC()->countries->get_states( $country ) : array(); + $formatted_state = ( $states && isset( $states[ $state ] ) ) ? $states[ $state ] : $state; + + return $formatted_state; +} + +function wc_gzd_get_shipment_order( $order ) { + if ( is_numeric( $order ) ) { + $order = wc_get_order( $order ); + } + + if ( is_a( $order, 'WC_Order' ) ) { + try { + return new Vendidero\Germanized\Shipments\Order( $order ); + } catch ( Exception $e ) { + wc_caught_exception( $e, __FUNCTION__, array( $order ) ); + return false; + } + } + + return false; +} + +function wc_gzd_get_shipment_label_title( $type, $plural = false ) { + $type_data = wc_gzd_get_shipment_type_data( $type ); + + return ( ! $plural ? $type_data['labels']['singular'] : $type_data['labels']['plural'] ); +} + +function wc_gzd_get_shipment_types() { + return array_keys( wc_gzd_get_shipment_type_data( false ) ); +} + +/** + * Get shipment type data by type. + * + * @param string $type type name. + * @return bool|array Details about the shipment type. + * + * @package Vendidero/Germanized/Shipments + */ +function wc_gzd_get_shipment_type_data( $type = false ) { + $types = apply_filters( + 'woocommerce_gzd_shipment_type_data', + array( + 'simple' => array( + 'class_name' => '\Vendidero\Germanized\Shipments\SimpleShipment', + 'labels' => array( + 'singular' => _x( 'Shipment', 'shipments', 'woocommerce-germanized' ), + 'plural' => _x( 'Shipments', 'shipments', 'woocommerce-germanized' ), + ), + ), + 'return' => array( + 'class_name' => '\Vendidero\Germanized\Shipments\ReturnShipment', + 'labels' => array( + 'singular' => _x( 'Return', 'shipments', 'woocommerce-germanized' ), + 'plural' => _x( 'Returns', 'shipments', 'woocommerce-germanized' ), + ), + ), + ) + ); + + if ( $type && array_key_exists( $type, $types ) ) { + return $types[ $type ]; + } elseif ( false === $type ) { + return $types; + } else { + return $types['simple']; + } +} + +function wc_gzd_get_shipments_by_order( $order ) { + $shipments = array(); + + if ( $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + $shipments = $order_shipment->get_shipments(); + } + + return $shipments; +} + +function wc_gzd_get_shipment_order_shipping_statuses() { + $shipment_statuses = array( + 'gzd-not-shipped' => _x( 'Not shipped', 'shipments', 'woocommerce-germanized' ), + 'gzd-partially-shipped' => _x( 'Partially shipped', 'shipments', 'woocommerce-germanized' ), + 'gzd-shipped' => _x( 'Shipped', 'shipments', 'woocommerce-germanized' ), + 'gzd-partially-delivered' => _x( 'Partially delivered', 'shipments', 'woocommerce-germanized' ), + 'gzd-delivered' => _x( 'Delivered', 'shipments', 'woocommerce-germanized' ), + 'gzd-no-shipping-needed' => _x( 'No shipping needed', 'shipments', 'woocommerce-germanized' ), + ); + + /** + * Filter to adjust or add order shipping statuses. + * An order might retrieve a shipping status e.g. not shipped. + * + * @param array $shipment_statuses Available order shipping statuses. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_order_shipping_statuses', $shipment_statuses ); +} + +function wc_gzd_get_shipment_order_return_statuses() { + $shipment_statuses = array( + 'gzd-open' => _x( 'Open', 'shipments', 'woocommerce-germanized' ), + 'gzd-partially-returned' => _x( 'Partially returned', 'shipments', 'woocommerce-germanized' ), + 'gzd-returned' => _x( 'Returned', 'shipments', 'woocommerce-germanized' ), + ); + + /** + * Filter to adjust or add order return statuses. + * An order might retrieve a shipping status e.g. not shipped. + * + * @param array $shipment_statuses Available order return statuses. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_order_return_statuses', $shipment_statuses ); +} + +/** + * @param $instance_id + * + * @return ShippingProvider\Method + */ +function wc_gzd_get_shipping_provider_method( $instance_id ) { + $original_id = $instance_id; + $method = false; + $method_id = ''; + + if ( is_a( $original_id, 'WC_Shipping_Rate' ) ) { + $instance_id = $original_id->get_instance_id(); + $method_id = $original_id->get_method_id(); + } elseif ( is_a( $original_id, 'WC_Shipping_Method' ) ) { + $instance_id = $original_id->get_instance_id(); + $method_id = $original_id->id; + } elseif ( ! is_numeric( $instance_id ) && is_string( $instance_id ) ) { + if ( strpos( $instance_id, ':' ) !== false ) { + $expl = explode( ':', $instance_id ); + $instance_id = ( ( ! empty( $expl ) && count( $expl ) > 1 ) ? (int) $expl[1] : 0 ); + $method_id = ( ! empty( $expl ) ) ? $expl[0] : $instance_id; + } else { + /** + * Plugins like Flexible Shipping use underscores to separate instance ids. + * Example: flexible_shipping_4_1. In this case, 4 ist the instance id. + * method_id: flexible_shipping + * instance_id: 4 + * + * On the other hand legacy shipping methods may be string only, e.g. an instance id might not exist. + * Example: local_pickup_plus + * method: local_pickup_plus + * instance_id: 0 + */ + $expl = explode( '_', $instance_id ); + $numbers = array_values( array_filter( $expl, 'is_numeric' ) ); + $method_id = rtrim( preg_replace( '/[0-9]+/', '', $instance_id ), '_' ); + + if ( ! empty( $numbers ) ) { + $instance_id = absint( $numbers[0] ); + } else { + $instance_id = 0; + } + } + } + + if ( ! empty( $instance_id ) ) { + // Make sure shipping zones are loaded + include_once WC_ABSPATH . 'includes/class-wc-shipping-zones.php'; + + /** + * Cache methods within frontend + */ + if ( WC()->session && did_action( 'woocommerce_shipping_init' ) ) { + $cache_key = 'woocommerce_gzd_method_' . $instance_id; + $tmp_method = WC()->session->get( $cache_key ); + + if ( ! $tmp_method || ! is_object( $tmp_method ) || is_a( $tmp_method, '__PHP_Incomplete_Class' ) ) { + $method = WC_Shipping_Zones::get_shipping_method( $instance_id ); + + if ( $method ) { + WC()->session->set( $cache_key, $method ); + } + } else { + $method = $tmp_method; + } + } else { + $method = WC_Shipping_Zones::get_shipping_method( $instance_id ); + } + } + + /** + * Fallback for legacy shipping methods that do not support instance ids. + */ + if ( ! $method && empty( $instance_id ) && ! empty( $method_id ) ) { + $shipping_methods = WC()->shipping()->get_shipping_methods(); + + if ( array_key_exists( $method_id, $shipping_methods ) ) { + $method = $shipping_methods[ $method_id ]; + } + } + + if ( $method ) { + /** + * Filter to adjust the classname used to construct the shipping provider method + * which contains additional provider related settings useful for shipments. + * + * @param string $classname The classname. + * @param WC_Shipping_Method $method The shipping method instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + $classname = apply_filters( 'woocommerce_gzd_shipping_provider_method_classname', 'Vendidero\Germanized\Shipments\ShippingProvider\Method', $method ); + + return new $classname( $method ); + } + + // Load placeholder + $placeholder = new ShippingProvider\MethodPlaceholder( $original_id ); + + /** + * Filter to adjust the fallback shipping method to be loaded if no real + * shipping method was able to be constructed (e.g. a custom plugin is being used which + * replaces the default Woo shipping zones integration). + * + * @param ShippingProvider\MethodPlaceholder $placeholder The placeholder impl. + * @param string $original_id The shipping method id. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipping_provider_method_fallback', $placeholder, $original_id ); +} + +/** + * Returns the current shipping method rate id. + * + * @return false|string + */ +function wc_gzd_get_current_shipping_method_id() { + $chosen_shipping_methods = WC()->session ? WC()->session->get( 'chosen_shipping_methods' ) : array(); + + if ( ! empty( $chosen_shipping_methods ) ) { + return reset( $chosen_shipping_methods ); + } + + return false; +} + +function wc_gzd_get_current_shipping_provider_method() { + if ( $current = wc_gzd_get_current_shipping_method_id() ) { + return wc_gzd_get_shipping_provider_method( $current ); + } + + return false; +} + +function wc_gzd_get_shipment_order_shipping_status_name( $status ) { + if ( 'gzd-' !== substr( $status, 0, 4 ) ) { + $status = 'gzd-' . $status; + } + + $status_name = ''; + $statuses = wc_gzd_get_shipment_order_shipping_statuses(); + + if ( array_key_exists( $status, $statuses ) ) { + $status_name = $statuses[ $status ]; + } + + /** + * Filter to adjust the status name for a certain order shipping status. + * + * @see wc_gzd_get_shipment_order_shipping_statuses() + * + * @param string $status_name The status name. + * @param string $status The shipping status. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_order_shipping_status_name', $status_name, $status ); +} + +function wc_gzd_get_shipment_order_return_status_name( $status ) { + if ( 'gzd-' !== substr( $status, 0, 4 ) ) { + $status = 'gzd-' . $status; + } + + $status_name = ''; + $statuses = wc_gzd_get_shipment_order_return_statuses(); + + if ( array_key_exists( $status, $statuses ) ) { + $status_name = $statuses[ $status ]; + } + + /** + * Filter to adjust the status name for a certain order return status. + * + * @see wc_gzd_get_shipment_order_return_statuses() + * + * @param string $status_name The status name. + * @param string $status The return status. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_order_return_status_name', $status_name, $status ); +} + +/** + * Standard way of retrieving shipments based on certain parameters. + * + * @param array $args Array of args (above). + * + * @return Shipment[] The shipments found. + *@since 3.0.0 + */ +function wc_gzd_get_shipments( $args ) { + $query = new Vendidero\Germanized\Shipments\ShipmentQuery( $args ); + + return $query->get_shipments(); +} + +function wc_gzd_get_shipment_customer_visible_statuses( $shipment_type = 'simple' ) { + $statuses = array_keys( wc_gzd_get_shipment_statuses() ); + $statuses = array_diff( $statuses, array( 'gzd-draft' ) ); + + /** + * Filter to decide which shipment statuses should be visible to customers + * e.g. whether a shipment of a certain status should be shown or not. + * + * @param array $shipment_statuses The available shipment statuses. + * @param string $shipment_type The shipment type. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_customer_visible_statuses', $statuses, $shipment_type ); +} + +/** + * Main function for returning shipments. + * + * @param mixed $the_shipment Object or shipment id. + * + * @return bool|SimpleShipment|ReturnShipment|Shipment + */ +function wc_gzd_get_shipment( $the_shipment ) { + return ShipmentFactory::get_shipment( $the_shipment ); +} + +/** + * Get all shipment statuses. + * + * @return array + */ +function wc_gzd_get_shipment_statuses() { + $shipment_statuses = array( + 'gzd-draft' => _x( 'Draft', 'shipments', 'woocommerce-germanized' ), + 'gzd-processing' => _x( 'Processing', 'shipments', 'woocommerce-germanized' ), + 'gzd-shipped' => _x( 'Shipped', 'shipments', 'woocommerce-germanized' ), + 'gzd-delivered' => _x( 'Delivered', 'shipments', 'woocommerce-germanized' ), + 'gzd-requested' => _x( 'Requested', 'shipments', 'woocommerce-germanized' ), + ); + + /** + * Add or adjust available Shipment statuses. + * + * @param array $shipment_statuses The available shipment statuses. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_statuses', $shipment_statuses ); +} + +/** + * @param Shipment $shipment + * + * @return mixed|void + */ +function wc_gzd_get_shipment_selectable_statuses( $shipment ) { + $shipment_statuses = wc_gzd_get_shipment_statuses(); + + if ( ! $shipment->has_status( 'requested' ) && isset( $shipment_statuses['gzd-requested'] ) ) { + unset( $shipment_statuses['gzd-requested'] ); + } + + /** + * Add or remove selectable shipment statuses for a certain shipment and/or shipment type. + * + * @param array $shipment_statuses The available shipment statuses. + * @param string $type The shipment type e.g. return. + * @param Shipment $shipment The shipment instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_selectable_statuses', $shipment_statuses, $shipment->get_type(), $shipment ); +} + +/** + * @param Order $order_shipment + * @param array $args + * + * @return ReturnShipment|WP_Error + */ +function wc_gzd_create_return_shipment( $order_shipment, $args = array() ) { + try { + + if ( ! $order_shipment || ! is_a( $order_shipment, 'Vendidero\Germanized\Shipments\Order' ) ) { + throw new Exception( _x( 'Invalid order.', 'shipments', 'woocommerce-germanized' ) ); + } + + if ( ! $order_shipment->needs_return() ) { + throw new Exception( _x( 'This order is already fully returned.', 'shipments', 'woocommerce-germanized' ) ); + } + + $args = wp_parse_args( + $args, + array( + 'items' => array(), + 'props' => array(), + ) + ); + + $shipment = ShipmentFactory::get_shipment( false, 'return' ); + + if ( ! $shipment ) { + throw new Exception( _x( 'Error while creating the shipment instance', 'shipments', 'woocommerce-germanized' ) ); + } + + // Make sure shipment knows its parent + $shipment->set_order_shipment( $order_shipment ); + $shipment->sync( $args['props'] ); + $shipment->sync_items( $args ); + $shipment->save(); + + } catch ( Exception $e ) { + return new WP_Error( 'error', $e->getMessage() ); + } + + return $shipment; +} + +/** + * @param Order $order_shipment + * @param array $args + * + * @return Shipment|WP_Error + */ +function wc_gzd_create_shipment( $order_shipment, $args = array() ) { + try { + if ( ! $order_shipment || ! is_a( $order_shipment, 'Vendidero\Germanized\Shipments\Order' ) ) { + throw new Exception( _x( 'Invalid shipment order', 'shipments', 'woocommerce-germanized' ) ); + } + + if ( ! $order = $order_shipment->get_order() ) { + throw new Exception( _x( 'Invalid shipment order', 'shipments', 'woocommerce-germanized' ) ); + } + + $args = wp_parse_args( + $args, + array( + 'items' => array(), + 'props' => array(), + ) + ); + + $shipment = ShipmentFactory::get_shipment( false, 'simple' ); + + if ( ! $shipment ) { + throw new Exception( _x( 'Error while creating the shipment instance', 'shipments', 'woocommerce-germanized' ) ); + } + + $shipment->set_order_shipment( $order_shipment ); + $shipment->sync( $args['props'] ); + $shipment->sync_items( $args ); + $shipment->save(); + + } catch ( Exception $e ) { + return new WP_Error( 'error', $e->getMessage() ); + } + + return $shipment; +} + +function wc_gzd_create_shipment_item( $shipment, $order_item, $args = array() ) { + try { + + if ( ! $order_item || ! is_a( $order_item, 'WC_Order_Item' ) ) { + throw new Exception( _x( 'Invalid order item', 'shipments', 'woocommerce-germanized' ) ); + } + + $item = new Vendidero\Germanized\Shipments\ShipmentItem(); + + $item->set_order_item_id( $order_item->get_id() ); + $item->set_shipment( $shipment ); + $item->sync( $args ); + $item->save(); + + } catch ( Exception $e ) { + return new WP_Error( 'error', $e->getMessage() ); + } + + return $item; +} + +function wc_gzd_allow_customer_return_empty_return_reason( $order ) { + return apply_filters( 'woocommerce_gzd_allow_customer_return_empty_return_reason', true, $order ); +} + +/** + * @param bool $allow_none + * @param bool|WC_Order_Item $order_item + * + * @return ReturnReason[] + */ +function wc_gzd_get_return_shipment_reasons( $order_item = false ) { + $reasons = Package::get_setting( 'return_reasons' ); + + if ( ! is_array( $reasons ) ) { + $reasons = array(); + } else { + $reasons = array_filter( $reasons ); + } + + /** + * Filter that allows adjusting raw return reasons for a specific shipment (e.g. array containing reason data with code, reason and order). + * + * @param array $reasons Available return reasons. + * @param WC_Order_Item|false $order_item The order item object if available to further filter reasons. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + $reasons = apply_filters( 'woocommerce_gzd_return_shipment_reasons_raw', $reasons, $order_item ); + $instances = array(); + + foreach ( $reasons as $reason ) { + $instances[] = new ReturnReason( $reason ); + } + + usort( $instances, '_wc_gzd_sort_return_shipment_reasons' ); + + /** + * Filter that allows to adjust available return reasons for a specific shipment. + * + * @param ReturnReason[] $reasons Available return reasons. + * @param WC_Order_Item|false $order_item The order item object if available to further filter reasons. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_return_shipment_reasons', $instances, $order_item ); +} + +function wc_gzd_return_shipment_reason_exists( $maybe_reason, $shipment = false ) { + $reasons = wc_gzd_get_return_shipment_reasons( $shipment ); + $exists = false; + + foreach ( $reasons as $reason ) { + + if ( $reason->get_code() === $maybe_reason ) { + $exists = true; + break; + } + } + + return $exists; +} + +/** + * @param ReturnReason $a + * @param ReturnReason $b + */ +function _wc_gzd_sort_return_shipment_reasons( $a, $b ) { + if ( $a->get_order() === $b->get_order() ) { + return 0; + } elseif ( $a->get_order() > $b->get_order() ) { + return 1; + } else { + return -1; + } +} + +/** + * @param WP_Error $error + * + * @return bool + */ +function wc_gzd_shipment_wp_error_has_errors( $error ) { + if ( is_callable( array( $error, 'has_errors' ) ) ) { + return $error->has_errors(); + } else { + $errors = $error->errors; + + return ( ! empty( $errors ) ? true : false ); + } +} + +/** + * @param Shipment $shipment + * @param ShipmentItem $shipment_item + * @param array $args + * + * @return ShipmentReturnItem|WP_Error + */ +function wc_gzd_create_return_shipment_item( $shipment, $shipment_item, $args = array() ) { + + try { + + if ( ! $shipment_item || ! is_a( $shipment_item, '\Vendidero\Germanized\Shipments\ShipmentItem' ) ) { + throw new Exception( _x( 'Invalid shipment item', 'shipments', 'woocommerce-germanized' ) ); + } + + $item = new Vendidero\Germanized\Shipments\ShipmentReturnItem(); + $item->set_order_item_id( $shipment_item->get_order_item_id() ); + $item->set_shipment( $shipment ); + $item->sync( $args ); + $item->save(); + + } catch ( Exception $e ) { + return new WP_Error( 'error', $e->getMessage() ); + } + + return $item; +} + +function wc_gzd_get_shipment_editable_statuses() { + /** + * Filter that allows to adjust Shipment statuses which decide upon whether + * a Shipment is editable or not. + * + * @param array $statuses Statuses which should be considered as editable. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_editable_statuses', array( 'draft', 'requested', 'processing' ) ); +} + +function wc_gzd_split_shipment_street( $street_str ) { + $return = array( + 'street' => $street_str, + 'number' => '', + 'addition' => '', + 'addition_2' => '', + ); + + try { + $split = AddressSplitter::split_address( $street_str ); + + $return['street'] = $split['streetName']; + $return['number'] = $split['houseNumber']; + /** + * e.g. 5. OG + */ + $return['addition'] = isset( $split['additionToAddress2'] ) ? $split['additionToAddress2'] : ''; + /** + * E.g. details to the location prefixed to the street name + */ + $return['addition_2'] = isset( $split['additionToAddress1'] ) ? $split['additionToAddress1'] : ''; + } catch ( Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } + + return $return; +} + +function wc_gzd_get_shipping_providers() { + return ShippingProvider\Helper::instance()->get_shipping_providers(); +} + +function wc_gzd_get_shipping_provider( $name ) { + return ShippingProvider\Helper::instance()->get_shipping_provider( $name ); +} + +function wc_gzd_get_default_shipping_provider() { + $default = Package::get_setting( 'default_shipping_provider' ); + + /** + * Filter to adjust the default shipping provider used as a fallback for shipments + * for which no provider could be determined automatically (e.g. by the chosen shipping methid). + * + * @param string $title The shipping provider slug. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_default_shipping_provider', $default ); +} + +function wc_gzd_get_shipping_provider_select() { + $providers = wc_gzd_get_shipping_providers(); + $select = array( + '' => _x( 'None', 'shipments', 'woocommerce-germanized' ), + ); + + foreach ( $providers as $provider ) { + if ( ! $provider->is_activated() ) { + continue; + } + $select[ $provider->get_name() ] = $provider->get_title(); + } + + return $select; +} + +function wc_gzd_get_shipping_provider_title( $slug ) { + $providers = wc_gzd_get_shipping_providers(); + + if ( array_key_exists( $slug, $providers ) ) { + $title = $providers[ $slug ]->get_title(); + } else { + $title = $slug; + } + + /** + * Filter to adjust the title of a certain shipping provider e.g. DHL. + * + * @param string $title The shipping provider title. + * @param string $slug The shipping provider slug. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipping_provider_title', $title, $slug ); +} + +/** + * @param Shipment $shipment + */ +function wc_gzd_get_shipment_shipping_provider_title( $shipment ) { + $title = $shipment->get_shipping_provider_title(); + + if ( empty( $title ) ) { + $title = apply_filters( 'woocommerce_gzd_shipping_provider_unknown_title', _x( 'Unknown', 'shipments-shipping-provider', 'woocommerce-germanized' ) ); + } + + return $title; +} + +function wc_gzd_get_shipping_provider_slug( $provider ) { + $providers = wc_gzd_get_shipping_providers(); + + if ( in_array( $provider, $providers, true ) ) { + $slug = array_search( $provider, $providers, true ); + } elseif ( array_key_exists( $provider, $providers ) ) { + $slug = $provider; + } else { + $slug = sanitize_key( $provider ); + } + + return $slug; +} + +function _wc_gzd_shipments_keep_force_filename( $new_filename ) { + return isset( $GLOBALS['gzd_shipments_unique_filename'] ) ? $GLOBALS['gzd_shipments_unique_filename'] : $new_filename; +} + +function wc_gzd_shipments_upload_data( $filename, $bits, $relative = true ) { + try { + Package::set_upload_dir_filter(); + $GLOBALS['gzd_shipments_unique_filename'] = $filename; + add_filter( 'wp_unique_filename', '_wc_gzd_shipments_keep_force_filename', 10, 1 ); + + $tmp = wp_upload_bits( $filename, null, $bits ); + + unset( $GLOBALS['gzd_shipments_unique_filename'] ); + remove_filter( 'wp_unique_filename', '_wc_gzd_shipments_keep_force_filename', 10 ); + Package::unset_upload_dir_filter(); + + if ( isset( $tmp['file'] ) ) { + $path = $tmp['file']; + + if ( $relative ) { + $path = Package::get_relative_upload_dir( $path ); + } + + return $path; + } else { + throw new Exception( _x( 'Error while uploading file.', 'shipments', 'woocommerce-germanized' ) ); + } + } catch ( Exception $e ) { + return false; + } +} + +function wc_gzd_get_shipment_setting_default_address_fields( $type = 'shipper' ) { + $address_fields = array( + 'first_name' => _x( 'First Name', 'shipments', 'woocommerce-germanized' ), + 'last_name' => _x( 'Last Name', 'shipments', 'woocommerce-germanized' ), + 'full_name' => _x( 'Full Name', 'shipments', 'woocommerce-germanized' ), + 'company' => _x( 'Company', 'shipments', 'woocommerce-germanized' ), + 'address_1' => _x( 'Address 1', 'shipments', 'woocommerce-germanized' ), + 'address_2' => _x( 'Address 2', 'shipments', 'woocommerce-germanized' ), + 'street' => _x( 'Street', 'shipments', 'woocommerce-germanized' ), + 'street_number' => _x( 'House Number', 'shipments', 'woocommerce-germanized' ), + 'postcode' => _x( 'Postcode', 'shipments', 'woocommerce-germanized' ), + 'city' => _x( 'City', 'shipments', 'woocommerce-germanized' ), + 'country' => _x( 'Country', 'shipments', 'woocommerce-germanized' ), + 'state' => _x( 'State', 'shipments', 'woocommerce-germanized' ), + 'phone' => _x( 'Phone', 'shipments', 'woocommerce-germanized' ), + 'email' => _x( 'Email', 'shipments', 'woocommerce-germanized' ), + 'customs_reference_number' => _x( 'Customs Reference Number', 'shipments', 'woocommerce-germanized' ), + 'customs_uk_vat_id' => _x( 'UK VAT ID (HMRC)', 'shipments', 'woocommerce-germanized' ), + ); + + return apply_filters( 'woocommerce_gzd_shipment_default_address_fields', $address_fields, $type ); +} + +/** + * @return array + */ +function wc_gzd_get_shipment_setting_address_fields( $address_type = 'shipper' ) { + $default_address_fields = array_keys( wc_gzd_get_shipment_setting_default_address_fields( $address_type ) ); + $default_address_data = array(); + + if ( 'return' === $address_type ) { + $default_address_data = wc_gzd_get_shipment_setting_address_fields( 'shipper' ); + } + + foreach ( $default_address_fields as $prop ) { + $key = "woocommerce_gzd_shipments_{$address_type}_address_{$prop}"; + $value = get_option( $key, '' ); + + if ( '' === $value && array_key_exists( $prop, $default_address_data ) ) { + $value = $default_address_data[ $prop ]; + } + + $address_fields[ $prop ] = $value; + } + + if ( ! empty( $address_fields['country'] ) && strlen( $address_fields['country'] ) > 2 ) { + $value = wc_format_country_state_string( $address_fields['country'] ); + $address_fields['country'] = $value['country']; + $address_fields['state'] = $value['state']; + } + + /** + * Format/split address 1 into street and house number + */ + if ( ! empty( $address_fields['address_1'] ) ) { + $split = wc_gzd_split_shipment_street( $address_fields['address_1'] ); + + $address_fields['street'] = $split['street']; + $address_fields['street_number'] = $split['number']; + } else { + $address_fields['street'] = ''; + $address_fields['street_number'] = ''; + } + + /** + * Attach formatted full name + */ + $address_fields['full_name'] = trim( sprintf( _x( '%1$s %2$s', 'full name', 'woocommerce-germanized' ), $address_fields['first_name'], $address_fields['last_name'] ) ); + + return apply_filters( "woocommerce_gzd_shipment_{$address_type}_address_fields", $address_fields, $address_type ); +} + +/** + * @param Order $shipment_order + * + * @return array + */ +function wc_gzd_get_shipment_return_address( $shipment_order = false ) { + return wc_gzd_get_shipment_setting_address_fields( 'return' ); +} + +/** + * @param WC_Order $order + */ +function wc_gzd_get_shipment_order_shipping_method_id( $order ) { + $methods = $order->get_shipping_methods(); + $id = ''; + + if ( ! empty( $methods ) ) { + $method_vals = array_values( $methods ); + $method = array_shift( $method_vals ); + + if ( $method ) { + $id = $method->get_method_id() . ':' . $method->get_instance_id(); + } + } + + /** + * Allows adjusting the shipping method id for a certain Order. + * + * @param string $id The shipping method id. + * @param WC_Order $order The order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_shipping_method_id', $id, $order ); +} + +function wc_gzd_render_shipment_action_buttons( $actions ) { + $actions_html = ''; + + foreach ( $actions as $action ) { + if ( isset( $action['group'] ) ) { + $actions_html .= '
' . wc_gzd_render_shipment_action_buttons( $action['actions'] ) . '
'; + } elseif ( isset( $action['action'], $action['url'], $action['name'] ) ) { + $target = isset( $action['target'] ) ? $action['target'] : '_self'; + + $actions_html .= sprintf( '%5$s', esc_attr( $action['action'] ), esc_url( $action['url'] ), esc_attr( isset( $action['title'] ) ? $action['title'] : $action['name'] ), $target, esc_html( $action['name'] ) ); + } + } + + return $actions_html; +} + +function wc_gzd_get_shipment_status_name( $status ) { + if ( 'gzd-' !== substr( $status, 0, 4 ) ) { + $status = 'gzd-' . $status; + } + + $status_name = ''; + $statuses = wc_gzd_get_shipment_statuses(); + + if ( array_key_exists( $status, $statuses ) ) { + $status_name = $statuses[ $status ]; + } + + /** + * Filter to adjust the shipment status name or title. + * + * @param string $status_name The status name or title. + * @param integer $status The status slug. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_status_name', $status_name, $status ); +} + +function wc_gzd_get_shipment_sent_statuses() { + /** + * Filter to adjust which Shipment statuses should be considered as sent. + * + * @param array $statuses An array of statuses considered as shipped, + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( + 'woocommerce_gzd_shipment_sent_statuses', + array( + 'shipped', + 'delivered', + ) + ); +} + +function wc_gzd_get_shipment_counts( $type = '' ) { + $counts = array(); + + foreach ( array_keys( wc_gzd_get_shipment_statuses() ) as $status ) { + $counts[ $status ] = wc_gzd_get_shipment_count( $status, $type ); + } + + return $counts; +} + +function wc_gzd_get_shipment_count( $status, $type = '' ) { + $count = 0; + $status = ( substr( $status, 0, 4 ) ) === 'gzd-' ? $status : 'gzd-' . $status; + $shipment_statuses = array_keys( wc_gzd_get_shipment_statuses() ); + + if ( ! in_array( $status, $shipment_statuses, true ) ) { + return 0; + } + + $cache_key = WC_Cache_Helper::get_cache_prefix( 'shipments' ) . $status . $type; + $cached_count = wp_cache_get( $cache_key, 'counts' ); + + if ( false !== $cached_count ) { + return $cached_count; + } + + $data_store = WC_Data_Store::load( 'shipment' ); + + if ( $data_store ) { + $count += $data_store->get_shipment_count( $status, $type ); + } + + wp_cache_set( $cache_key, $count, 'counts' ); + + return $count; +} + +/** + * See if a string is a shipment status. + * + * @param string $maybe_status Status, including any gzd- prefix. + * @return bool + */ +function wc_gzd_is_shipment_status( $maybe_status ) { + $shipment_statuses = wc_gzd_get_shipment_statuses(); + + return isset( $shipment_statuses[ $maybe_status ] ); +} + +/** + * Main function for returning shipment items. + * + * @since 2.2 + * + * @param mixed $the_item Object or shipment item id. + * @param string $item_type The shipment item type. + * + * @return bool|ShipmentItem + */ +function wc_gzd_get_shipment_item( $the_item = false, $item_type = 'simple' ) { + $item_id = wc_gzd_get_shipment_item_id( $the_item ); + + if ( ! $item_id ) { + return false; + } + + $item_class = 'Vendidero\Germanized\Shipments\ShipmentItem'; + + if ( 'return' === $item_type ) { + $item_class = 'Vendidero\Germanized\Shipments\ShipmentReturnItem'; + } + + /** + * Filter to adjust the classname used to construct a ShipmentItem. + * + * @param string $classname The classname to be used. + * @param integer $item_id The shipment item id. + * @param string $item_type The shipment item type. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $classname = apply_filters( 'woocommerce_gzd_shipment_item_class', $item_class, $item_id, $item_type ); + + if ( ! class_exists( $classname ) ) { + return false; + } + + try { + return new $classname( $item_id ); + } catch ( Exception $e ) { + wc_caught_exception( $e, __FUNCTION__, array( $the_item, $item_type ) ); + return false; + } +} + +/** + * Get the shipment item ID depending on what was passed. + * + * @since 3.0.0 + * @param mixed $item Item data to convert to an ID. + * @return int|bool false on failure + */ +function wc_gzd_get_shipment_item_id( $item ) { + if ( is_numeric( $item ) ) { + return $item; + } elseif ( $item instanceof Vendidero\Germanized\Shipments\ShipmentItem ) { + return $item->get_id(); + } elseif ( ! empty( $item->shipment_item_id ) ) { + return $item->shipment_item_id; + } else { + return false; + } +} + +/** + * Format dimensions for display. + * + * @since 3.0.0 + * @param array $dimensions Array of dimensions. + * @return string + */ +function wc_gzd_format_shipment_dimensions( $dimensions, $unit = '' ) { + $dimension_string = implode( ' × ', array_filter( array_map( 'wc_format_localized_decimal', $dimensions ) ) ); + + if ( ! empty( $dimension_string ) ) { + $unit = empty( $unit ) ? get_option( 'woocommerce_dimension_unit' ) : $unit; + $dimension_string .= ' ' . $unit; + } else { + $dimension_string = _x( 'N/A', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Filter to adjust the format of Shipment dimensions e.g. LxBxH. + * + * @param string $dimension_string The dimension string. + * @param array $dimensions Array containing the dimensions. + * @param string $unit The dimension unit. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_format_shipment_dimensions', $dimension_string, $dimensions, $unit ); +} + +/** + * Format a weight for display. + * + * @since 3.0.0 + * @param float $weight Weight. + * @return string + */ +function wc_gzd_format_shipment_weight( $weight, $unit = '' ) { + $weight_string = wc_format_localized_decimal( $weight ); + + if ( ! empty( $weight_string ) ) { + $unit = empty( $unit ) ? get_option( 'woocommerce_weight_unit' ) : $unit; + $weight_string .= ' ' . $unit; + } else { + $weight_string = _x( 'N/A', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Filter to adjust the format of Shipment weight. + * + * @param string $weight_string The weight string. + * @param string $weight The Shipment weight. + * @param string $unit The dimension unit. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_format_shipment_weight', $weight_string, $weight, $unit ); +} + +/** + * Get My Account > Shipments columns. + * + * @since 3.0.0 + * @return array + */ +function wc_gzd_get_account_shipments_columns( $type = 'simple' ) { + /** + * Filter to adjust columns being used to display shipments in a table view on the customer + * account page. + * + * @param string[] $columns The columns in key => value pairs. + * @param string $type The shipment type e.g. simple or return. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $columns = apply_filters( + 'woocommerce_gzd_account_shipments_columns', + array( + 'shipment-number' => _x( 'Shipment', 'shipments', 'woocommerce-germanized' ), + 'shipment-date' => _x( 'Date', 'shipments', 'woocommerce-germanized' ), + 'shipment-status' => _x( 'Status', 'shipments', 'woocommerce-germanized' ), + 'shipment-tracking' => _x( 'Tracking', 'shipments', 'woocommerce-germanized' ), + 'shipment-actions' => _x( 'Actions', 'shipments', 'woocommerce-germanized' ), + ), + $type + ); + + return $columns; +} + +function wc_gzd_get_order_customer_add_return_url( $order ) { + + if ( ! $shipment_order = wc_gzd_get_shipment_order( $order ) ) { + return false; + } + + $url = wc_get_endpoint_url( 'add-return-shipment', $shipment_order->get_order()->get_id(), wc_get_page_permalink( 'myaccount' ) ); + + if ( $shipment_order->get_order()->get_customer_id() <= 0 ) { + $key = $shipment_order->get_order_return_request_key(); + + if ( ! empty( $key ) ) { + $url = add_query_arg( array( 'key' => $key ), $url ); + } else { + $url = ''; + } + } + + /** + * Filter to adjust the URL the customer (or guest) might access to add a return to a certain order. + * + * @param string $url The URL pointing to the add return page. + * @param Order $order The order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipments_add_return_shipment_url', $url, $shipment_order->get_order() ); +} + +/** + * @param WC_Order $order + * + * @return mixed + */ +function wc_gzd_order_is_customer_returnable( $order, $check_date = true ) { + $is_returnable = false; + + if ( ! $shipment_order = wc_gzd_get_shipment_order( $order ) ) { + return false; + } + + if ( $provider = wc_gzd_get_order_shipping_provider( $order ) ) { + $is_returnable = $provider->supports_customer_returns( $shipment_order->get_order() ); + + if ( $shipment_order->get_order()->get_customer_id() <= 0 && ! $provider->supports_guest_returns() ) { + $is_returnable = false; + } + } + + // Shipment is fully returned + if ( ! $shipment_order->needs_return() ) { + $is_returnable = false; + } + + // Check days left for return + $maximum_days = Package::get_setting( 'customer_return_open_days' ); + + if ( $check_date && ! empty( $maximum_days ) ) { + $maximum_days = absint( $maximum_days ); + + if ( ! empty( $maximum_days ) ) { + + $completed_date = $shipment_order->get_order()->get_date_created(); + + if ( $shipment_order->get_date_shipped() ) { + $completed_date = $shipment_order->get_date_shipped(); + } elseif ( $shipment_order->get_order()->get_date_completed() ) { + $completed_date = $shipment_order->get_order()->get_date_completed(); + } + + /** + * Filter to adjust the completed date of an order used to determine whether an order is + * still returnable by the customer or not. The date is constructed by checking for existence in the following order: + * + * 1. The date the order was shipped completely + * 2. The date the order was marked as completed + * 3. The date the order was created + * + * @param WC_DateTime $completed_date The order completed date. + * @param WC_Order $order The order instance. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + $completed_date = apply_filters( 'woocommerce_gzd_order_return_completed_date', $completed_date, $shipment_order->get_order() ); + + if ( $completed_date ) { + $today = new WC_DateTime(); + $diff = $today->diff( $completed_date ); + + if ( $diff->days > $maximum_days ) { + $is_returnable = false; + } + } + } + } + + /** + * Filter to decide whether a customer might add return request to a certain order. + * + * @param bool $is_returnable Whether or not shipment supports customer added returns + * @param WC_Order $order The order instance for which the return shall be created. + * @param bool $check_date Whether to check for a maximum date or not. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_order_is_returnable_by_customer', $is_returnable, $shipment_order->get_order(), $check_date ); +} + +/** + * @param $order + * + * @return bool|\Vendidero\Germanized\Shipments\Interfaces\ShippingProvider + */ +function wc_gzd_get_order_shipping_provider( $order ) { + if ( is_numeric( $order ) ) { + $order = wc_get_order( $order ); + } + + if ( ! $order ) { + return false; + } + + $provider = false; + + if ( $method = wc_gzd_get_shipping_provider_method( wc_gzd_get_shipment_order_shipping_method_id( $order ) ) ) { + $provider = $method->get_provider_instance(); + } + + /** + * Filters the shipping provider detected for a specific order. + * + * @param bool|\Vendidero\Germanized\Shipments\Interfaces\ShippingProvider $provider The shipping provider instance. + * @param WC_Order $order The order instance. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_get_order_shipping_provider', $provider, $order ); +} + +function wc_gzd_get_customer_order_return_request_key() { + $key = ( isset( $_REQUEST['key'] ) ? wc_clean( wp_unslash( $_REQUEST['key'] ) ) : '' ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + return $key; +} + +function wc_gzd_customer_can_add_return_shipment( $order_id ) { + $can_view_shipments = false; + + if ( isset( $_REQUEST['key'] ) && ! empty( $_REQUEST['key'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $key = wc_gzd_get_customer_order_return_request_key(); + + if ( ( $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) && ! empty( $key ) ) { + + if ( hash_equals( $order_shipment->get_order_return_request_key(), $key ) ) { + $can_view_shipments = true; + } + } + } elseif ( is_user_logged_in() ) { + $can_view_shipments = current_user_can( 'view_order', $order_id ); + } + + /** + * Filters whether a logged in user (or guest) might view shipments belonging to an order or not. + * + * @param bool $can_view_shipments Whether the user (or guest) might see shipments or not. + * @param integer $order_id The order id. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_customer_can_view_shipments', $can_view_shipments, $order_id ); +} + +/** + * @param WC_Order|integer $order + */ +function wc_gzd_customer_return_needs_manual_confirmation( $order ) { + if ( is_numeric( $order ) ) { + $order = wc_get_order( $order ); + } + + if ( ! $order ) { + return true; + } + + $needs_manual_confirmation = true; + + if ( $provider = wc_gzd_get_order_shipping_provider( $order ) ) { + $needs_manual_confirmation = $provider->needs_manual_confirmation_for_returns(); + } + + /** + * Filter to decide whether a customer added return of a certain order + * needs manual confirmation by the shop manager or not. + * + * @param bool $needs_manual_confirmation Whether needs manual confirmation or not. + * @param WC_Order $order The order instance for which the return shall be created. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_customer_return_needs_manual_confirmation', $needs_manual_confirmation, $order ); +} + +/** + * Get account shipments actions. + * + * @since 3.2.0 + * @param int|Shipment $shipment Shipment instance or ID. + * @return array + */ +function wc_gzd_get_account_shipments_actions( $shipment ) { + + if ( ! is_object( $shipment ) ) { + $shipment_id = absint( $shipment ); + $shipment = wc_gzd_get_shipment( $shipment_id ); + } + + if ( ! $shipment ) { + return array(); + } + + $actions = array( + 'view' => array( + 'url' => $shipment->get_view_shipment_url(), + 'name' => _x( 'View', 'shipments', 'woocommerce-germanized' ), + ), + ); + + if ( 'return' === $shipment->get_type() && $shipment->has_label() && ! $shipment->has_status( 'delivered' ) ) { + $actions['download-label'] = array( + 'url' => $shipment->get_label()->get_download_url(), + 'name' => _x( 'Download label', 'shipments', 'woocommerce-germanized' ), + ); + } + + /** + * Filter to adjust available actions in the shipments table view on the customer account page + * for a specific shipment. + * + * @param string[] $actions Available actions containing an id as key and a URL and name. + * @param Shipment $shipment The shipment instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_account_shipments_actions', $actions, $shipment ); +} + +function wc_gzd_shipments_get_product( $the_product ) { + try { + if ( is_a( $the_product, '\Vendidero\Germanized\Shipments\Product' ) ) { + return $the_product; + } + + return new \Vendidero\Germanized\Shipments\Product( $the_product ); + } catch ( \Exception $e ) { + return false; + } +} + +function wc_gzd_get_volume_dimension( $dimension, $to_unit, $from_unit = '' ) { + $to_unit = strtolower( $to_unit ); + + if ( empty( $from_unit ) ) { + $from_unit = strtolower( get_option( 'woocommerce_dimension_unit' ) ); + } + + // Unify all units to cm first. + if ( $from_unit !== $to_unit ) { + switch ( $from_unit ) { + case 'm': + $dimension *= 1000000; + break; + case 'mm': + $dimension *= 0.001; + break; + } + + // Output desired unit. + switch ( $to_unit ) { + case 'm': + $dimension *= 0.000001; + break; + case 'mm': + $dimension *= 1000; + break; + } + } + + return ( $dimension < 0 ) ? 0 : $dimension; +} + +if ( ! function_exists( 'wc_gzd_wp_theme_get_element_class_name' ) ) { + /** + * Given an element name, returns a class name. + * + * If the WP-related function is not defined, return empty string. + * + * @param string $element The name of the element. + * + * @return string + */ + function wc_gzd_wp_theme_get_element_class_name( $element ) { + if ( function_exists( 'wc_wp_theme_get_element_class_name' ) ) { + return wc_wp_theme_get_element_class_name( $element ); + } elseif ( function_exists( 'wp_theme_get_element_class_name' ) ) { + return wp_theme_get_element_class_name( $element ); + } + + return ''; + } +} + +/** + * Forces a WP_Error object to be converted to a ShipmentError. + * + * @param $error + * + * @return mixed|\Vendidero\Germanized\Shipments\ShipmentError + */ +function wc_gzd_get_shipment_error( $error ) { + if ( ! is_wp_error( $error ) ) { + return $error; + } elseif ( is_a( $error, 'Vendidero\Germanized\Shipments\ShipmentError' ) ) { + return $error; + } else { + return \Vendidero\Germanized\Shipments\ShipmentError::from_wp_error( $error ); + } +} diff --git a/packages/woocommerce-germanized-shipments/includes/wc-gzd-shipment-template-hooks.php b/packages/woocommerce-germanized-shipments/includes/wc-gzd-shipment-template-hooks.php new file mode 100644 index 000000000..3f211f474 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/includes/wc-gzd-shipment-template-hooks.php @@ -0,0 +1,28 @@ + false, + 'show_image' => false, + 'image_size' => array( 32, 32 ), + 'plain_text' => false, + 'sent_to_admin' => false, + ); + + $args = wp_parse_args( $args, $defaults ); + $template = $args['plain_text'] ? 'emails/plain/email-shipment-items.php' : 'emails/email-shipment-items.php'; + + wc_get_template( + $template, + /** + * Filter to adjust the arguments passed to retrieving ShipmentItems for display in an Email. + * + * @param array $args Array containing the arguments passed. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + apply_filters( + 'woocommerce_gzd_email_shipment_items_args', + array( + 'shipment' => $shipment, + 'items' => $shipment->get_items(), + 'show_sku' => $args['show_sku'], + 'show_image' => $args['show_image'], + 'image_size' => $args['image_size'], + 'plain_text' => $args['plain_text'], + 'sent_to_admin' => $args['sent_to_admin'], + ) + ) + ); + + /** + * Filter that allows adjusting the HTML output of the shipment email item table. + * + * @param string $html The HTML output. + * @param Shipment $shipment The shipment instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_email_shipment_items_table', ob_get_clean(), $shipment ); + } +} + +if ( ! function_exists( 'woocommerce_gzd_shipments_template_view_shipments' ) ) { + + function woocommerce_gzd_shipments_template_view_shipments( $order_id ) { + + if ( ( ! ( $order = wc_get_order( $order_id ) ) ) || ( ! current_user_can( 'view_order', $order_id ) ) ) { + echo '
' . esc_html_x( 'Invalid order.', 'shipments', 'woocommerce-germanized' ) . ' ' . esc_html_x( 'My account', 'shipments', 'woocommerce-germanized' ) . '
'; + + return; + } + + $shipments = wc_gzd_get_shipments( + array( + 'order_id' => $order_id, + 'type' => 'simple', + 'status' => wc_gzd_get_shipment_customer_visible_statuses(), + ) + ); + + $returns = wc_gzd_get_shipments( + array( + 'order_id' => $order_id, + 'type' => 'return', + 'status' => wc_gzd_get_shipment_customer_visible_statuses( 'return' ), + ) + ); + + wc_get_template( + 'myaccount/order-shipments.php', + array( + 'order_id' => $order_id, + 'order' => $order, + 'shipments' => $shipments, + 'returns' => $returns, + ) + ); + } +} + +if ( ! function_exists( 'woocommerce_gzd_shipments_template_view_shipment' ) ) { + + function woocommerce_gzd_shipments_template_view_shipment( $shipment_id ) { + + if ( ( ! ( $shipment = wc_gzd_get_shipment( $shipment_id ) ) ) || ( ! current_user_can( 'view_order', $shipment->get_order_id() ) ) ) { + echo '
' . esc_html_x( 'Invalid shipment.', 'shipments', 'woocommerce-germanized' ) . ' ' . esc_html_x( 'My account', 'shipments', 'woocommerce-germanized' ) . '
'; + + return; + } + + wc_get_template( + 'myaccount/view-shipment.php', + array( + 'shipment' => $shipment, + 'shipment_id' => $shipment_id, + ) + ); + } +} + +if ( ! function_exists( 'woocommerce_gzd_shipments_template_add_return_shipment' ) ) { + + function woocommerce_gzd_shipments_template_add_return_shipment( $order_id ) { + + if ( ( ! ( $order = wc_get_order( $order_id ) ) ) || ( ! wc_gzd_customer_can_add_return_shipment( $order_id ) ) ) { + echo '
' . esc_html_x( 'Invalid order.', 'shipments', 'woocommerce-germanized' ) . ' ' . esc_html_x( 'My account', 'shipments', 'woocommerce-germanized' ) . '
'; + return; + } + + if ( ! wc_gzd_order_is_customer_returnable( $order_id ) ) { + echo '
' . esc_html_x( 'Currently you cannot add new return requests to that order. If you have questions regarding the return of that order please contact us for further information.', 'shipments', 'woocommerce-germanized' ) . ( is_user_logged_in() ? ' ' . esc_html_x( 'View order', 'shipments', 'woocommerce-germanized' ) . '' : '' ) . '
'; + return; + } + + $notices = function_exists( 'wc_print_notices' ) ? wc_print_notices( true ) : ''; + + // Output notices in case notices have not been outputted yet. + if ( ! empty( $notices ) ) { + echo '
' . $notices . '
'; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + } + + wc_get_template( + 'myaccount/add-return-shipment.php', + array( + 'order' => $order, + 'order_id' => $order_id, + 'shipment_order' => wc_gzd_get_shipment_order( $order ), + ) + ); + } +} + +if ( ! function_exists( 'woocommerce_gzd_shipment_details_table' ) ) { + + function woocommerce_gzd_shipment_details_table( $shipment_id ) { + if ( ! $shipment_id ) { + return; + } + + wc_get_template( + 'shipment/shipment-details.php', + array( + 'shipment_id' => $shipment_id, + ) + ); + } +} + +if ( ! function_exists( 'woocommerce_gzd_return_shipments_template_instructions' ) ) { + + function woocommerce_gzd_return_shipments_template_instructions( $shipment_id ) { + if ( $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + + if ( 'return' !== $shipment->get_type() ) { + return; + } + + if ( ! $shipment->has_status( array( 'processing', 'sent' ) ) ) { + return; + } + + wc_get_template( + 'shipment/shipment-return-instructions.php', + array( + 'shipment' => $shipment, + 'shipment_id' => $shipment_id, + ) + ); + } + } +} diff --git a/packages/woocommerce-germanized-shipments/license.txt b/packages/woocommerce-germanized-shipments/license.txt new file mode 100644 index 000000000..16aa3cb39 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/license.txt @@ -0,0 +1,699 @@ +WooCommerce Germanized Shipments + +Copyright 2019 by the contributors + +This program is free software; you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation; either version 3 of the License, or +(at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with this program; if not, write to the Free Software +Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + +This program incorporates work covered by the following copyright and +permission notices: + + WooCommerce + +=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= + + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright © 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright © + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright © + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. \ No newline at end of file diff --git a/packages/woocommerce-germanized-shipments/src/AddressSplitter.php b/packages/woocommerce-germanized-shipments/src/AddressSplitter.php new file mode 100644 index 000000000..2b59c5e0e --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/AddressSplitter.php @@ -0,0 +1,272 @@ + + * + */ + $addition_2_introducers = '(?: + # {{{ Additions relating to who (a natural person) is addressed + \s+ [Cc] \s* \/ \s* [Oo] \s + | ℅ + | \s+ care \s+ of \s+ + # German, Swiss, Austrian + | \s+ (?: p|p.\s*|per\s+ ) (?: A|A.|Adr.|(?<=\s)Adresse ) \s + | \s+ p. \s* A. \s + | \s+ (?: z | z.\s* | zu\s+ ) (?: Hd|Hd.|(?<=\s)Händen|(?<=\s)Haenden|(?<=\s)Handen) \s+ + ## o. V. i. A. = oder Vertreter im Amt + | \s+ (?: o | o.\s* | oder\s+ ) + (?: V | V.\s* | (?<=\s)Vertreter\s+ ) + (?: i | i.\s* | (?<=\s)im\s+ ) + (?: A | A.\s* | (?<=\s)Amt\s+ ) + # }}} + # {{{ Additions which further specify more precisely the location + | \s+ (?: Haus ) \s + | \s+ (?: WG | W\.G\. | WG\. | Wohngemeinschaft ) ($ | \s) + | \s+ (?: [Aa]partment | APT \.? | Apt \.? ) \s + | \s+ (?: [Ff]lat ) \s + | (?: # Numeric-based location specifiers (e.g., "3. Stock"): + \s+ + (?: + [\p{N}]+ # A number, … + (?i: st | nd | rd | th)? # …, optionally followed by an English number suffix + \.? # …, followed by an optional dot, + \s* # …, followed by optional spacing + )? + (?: # Specifying category: + (?i: Stock | Stockwerk) + | App \.? | Apt \.? | (?i: Appartment | Apartment) + ) + # At the end of the string or followed by a space + (?: $ | \s) + ) + | (?: + \s+ (?: + # English language stop words wrt location from source [1] + # (extracted only those which may not be _exclusively_ part of + # street names): + | ANX \.? | (?i: ANNEX) + | APT \.? | (?i: APARTMENT) + | ARC \.? | (?i: ARCADE) + | AVE \.? | (?i: AVENUE) + | BSMT \.? | (?i: BASEMENT) + | BLDG \.? | (?i: BUILDING) + | CP \.? | (?i: CAMP) + | COR \.? | (?i: CORNER) + | CORS \.? | (?i: CORNERS) + | CT \.? | (?i: COURT) + | CTS \.? | (?i: COURTS) + | DEPT \.? | (?i: DEPARTMENT) + | DV \.? | (?i: DIVIDE) + | EST \.? | (?i: ESTATE) + | EXT \.? | (?i: EXTENSION) + | FRY \.? | (?i: FERRY) + | FLD \.? | (?i: FIELD) + | FLDS \.? | (?i: FIELDS) + | FLT \.? | (?i: FLAT) + | FL \.? | (?i: FLOOR) + | FRNT \.? | (?i: FRONT) + | GDNS \.? | (?i: GARDEN) + | GDNS \.? | (?i: GARDENS) + | GTWY \.? | (?i: GATEWAY) + | GRN \.? | (?i: GREEN) + | GRV \.? | (?i: GROVE) + | HNGR \.? | (?i: HANGER) + | HBR \.? | (?i: HARBOR) + | HVN \.? | (?i: HAVEN) + | KY \.? | (?i: KEY) + | LBBY \.? | (?i: LOBBY) + | LCKS \.? | (?i: LOCK) + | LCKS \.? | (?i: LOCKS) + | LDG \.? | (?i: LODGE) + | MNR \.? | (?i: MANOR) + | OFC \.? | (?i: OFFICE) + | PKWY \.? | (?i: PARKWAY) + | PH \.? | (?i: PENTHOUSE) + | PRT \.? | (?i: PORT) + | RADL \.? | (?i: RADIAL) + | RM \.? | (?i: ROOM) + | SPC \.? | (?i: SPACE) + | SQ \.? | (?i: SQUARE) + | STA \.? | (?i: STATION) + | STE \.? | (?i: SUITE) + | TER \.? | (?i: TERRACE) + | TRAK \.? | (?i: TRACK) + | TRL \.? | (?i: TRAIL) + | TRLR \.? | (?i: TRAILER) + | TUNL \.? | (?i: TUNNEL) + | VW \.? | (?i: VIEW) + | VIS \.? | (?i: VISTA) + # Custom custom additions: + | (?i: Story | Storey) + | LVL \.? | (?i: Level) + ) + # May optionally be followed directly by a number+letter + # combination (e.g., "LVL3C"): + (?: [\p{N}]+[\p{L}]* )? + # Occurs at the end of the string or followed by a space: + ($ | \s) + ) + # Heuristic to match location specifiers. These must not be + # conflated with house number extensions as in "12 AB". Hence + # our heuristic is at least 3 letters with the first letter being + # spelled as a capital. E.g., it would match "Haus", "Gebäude" or + # "Arbeitspl.", but not "AAB". + | \s+ ( [\p{Lu}\p{Lt}] [\p{Ll}\p{Lo}]{2,} \.? ) ($ | \s) + # }}} + )'; + $regex = ' + /\A\s* + (?: ######################################################################### + # Option A: [] # + # [] # + ######################################################################### + (?:(?P.*?),\s*)? # Addition to address 1 + (?:No\.\s*)? + (?P + (?P + \pN+(\s+\d+\/\d+)? + ) + (?: + \s*[\-\/\.]?\s* + (?P(?:[a-zA-Z\pN]){1,2}) + \s+ + )? + ) + \s*,?\s* + (?P(?:[a-zA-Z]\s*|\pN\pL{2,}\s\pL)\S[^,#]*?(?(?!\s).*?))? # Addition to address 2 + | ######################################################################### + # Option B: [] # + # [] # + ######################################################################### + (?:(?P.*?),\s*(?=.*[,\/]))? # Addition to address 1 + (?!\s*No\.)(?P[^0-9# ]\s*\S(?:[^,#](?!\b\pN+\s))*?(? + (?P + \pN+ + ) + (?: + # Match house numbers that are (optionally) amended + # by a dash (e.g., 12-13) or slash (e.g., 12\/A): + (?: \s*[\-\/]\s* )* + (?P + (?: + # Do not match "care-of"-like additions as + # house numbers: + (?!' . $addition_2_introducers . ') + \s*[\pL\pN]+ + ) + # Match any further slash- or dash-based house + # number extensions: + (?: + # Do not match "care-of"-like additions as + # house numbers: + (?!' . $addition_2_introducers . ') + # Match any (optionally space-separated) + # additionals parts of house numbers after + # slashes or dashes. + \s* [\-\/] \s* + [\pL\pN]+ + )* + ) + )? + ) # House number + (?: + (?:\s*[-,\/]|(?=\#)|\s)\s*(?!\s*No\.)\s* + (?P(?!\s).*?) + )? + ) + \s*\Z/xu'; + $result = preg_match( $regex, $address, $matches ); + if ( 0 === $result ) { + throw new Exception( sprintf( 'Error occurred while trying to split address \'%s\'', $address ) ); + } elseif ( false === $result ) { + throw new RuntimeException( sprintf( 'Error occurred while trying to split address \'%s\'', $address ) ); + } + if ( ! empty( $matches['A_Street_name'] ) ) { + return array( + 'additionToAddress1' => $matches['A_Addition_to_address_1'], + 'streetName' => $matches['A_Street_name'], + 'houseNumber' => $matches['A_House_number_match'], + 'houseNumberParts' => array( + 'base' => $matches['A_House_number_base'], + 'extension' => isset( $matches['A_House_number_extension'] ) ? trim( $matches['A_House_number_extension'] ) : '', + ), + 'additionToAddress2' => ( isset( $matches['A_Addition_to_address_2'] ) ) ? $matches['A_Addition_to_address_2'] : '', + ); + } else { + return array( + 'additionToAddress1' => $matches['B_Addition_to_address_1'], + 'streetName' => $matches['B_Street_name'], + 'houseNumber' => $matches['B_House_number_match'], + 'houseNumberParts' => array( + 'base' => $matches['B_House_number_base'], + 'extension' => isset( $matches['B_House_number_extension'] ) ? trim( $matches['B_House_number_extension'] ) : '', + ), + 'additionToAddress2' => isset( $matches['B_Addition_to_address_2'] ) ? $matches['B_Addition_to_address_2'] : '', + ); + } + } + + /** + * @param string $house_number A house number string to split in base and extension + * + * @return array + * @throws Exception + */ + public static function split_house_number( $house_number ) { + $regex = + '/ + \A\s* # Trim white spaces at the beginning + (?:[nN][oO][\.:]?\s*)? # Trim sth. like No. + (?:\#\s*)? # Trim # + (? + [\pN]+ # House Number base (only the number) + ) + \s*[\/\-\.]?\s* # Separator (optional) + (? # House number extension (optional) + .*? # Here we allow every character. Everything after the separator is interpreted as extension + ) + \s*\Z # Trim white spaces at the end + /xu'; // Option (u)nicode and e(x)tended syntax + $result = preg_match( $regex, $house_number, $matches ); + if ( 0 === $result ) { + throw new Exception( sprintf( 'Error occurred while trying to house number \'%s\'', $house_number ) ); + } elseif ( false === $result ) { + throw new RuntimeException( sprintf( 'Error occurred while trying to house number \'%s\'', $house_number ) ); + } + + return array( + 'base' => $matches['House_number_base'], + 'extension' => $matches['House_number_extension'], + ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/Admin.php b/packages/woocommerce-germanized-shipments/src/Admin/Admin.php new file mode 100644 index 000000000..388527f88 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/Admin.php @@ -0,0 +1,1191 @@ +get_id() !== $post_id ) { + $the_order = wc_get_order( $post_id ); + } + + if ( $shipment_order = wc_gzd_get_shipment_order( $the_order ) ) { + $shipping_status = $shipment_order->get_shipping_status(); + $status_html = '' . esc_html( wc_gzd_get_shipment_order_shipping_status_name( $shipping_status ) ) . ''; + + if ( in_array( $shipping_status, array( 'shipped', 'partially-shipped' ), true ) && $shipment_order->get_shipments() ) { + echo '' . wp_kses_post( $status_html ) . ''; + } else { + echo wp_kses_post( $status_html ); + } + } + } + } + + public static function register_order_shipping_status_column( $columns ) { + $new_columns = array(); + $added_column = false; + + foreach ( $columns as $column_name => $title ) { + if ( ! $added_column && ( 'shipping_address' === $column_name || 'wc_actions' === $column_name ) ) { + $new_columns['shipping_status'] = _x( 'Shipping Status', 'shipments-order-column-name', 'woocommerce-germanized' ); + $added_column = true; + } + + $new_columns[ $column_name ] = $title; + } + + if ( ! $added_column ) { + $new_columns['shipping_status'] = _x( 'Shipping Status', 'shipments-order-column-name', 'woocommerce-germanized' ); + } + + return $new_columns; + } + + /** + * In case the shipper/return country is set to AF (or DE with missing state) due to a bug in Woo, make sure + * to automatically adjust it to the right value in case the base country option is being saved. + * + * @return void + */ + public static function observe_base_country_setting() { + if ( isset( $_POST['woocommerce_default_country'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $new_base_country = wc_format_country_state_string( get_option( 'woocommerce_default_country' ) ); + + if ( 'AF' !== $new_base_country['country'] ) { + $shipper_country = wc_format_country_state_string( get_option( 'woocommerce_gzd_shipments_shipper_address_country' ) ); + $return_country = wc_format_country_state_string( get_option( 'woocommerce_gzd_shipments_return_address_country' ) ); + + if ( 'AF' === $shipper_country['country'] || ( 'DE' === $new_base_country['country'] && 'DE' === $shipper_country['country'] && empty( $shipper_country['state'] ) && ! empty( $new_base_country['state'] ) ) ) { + update_option( 'woocommerce_gzd_shipments_shipper_address_country', get_option( 'woocommerce_default_country' ) ); + } + + if ( 'AF' === $return_country['country'] || ( 'DE' === $new_base_country['country'] && 'DE' === $return_country['country'] && empty( $return_country['state'] ) && ! empty( $return_country['state'] ) ) ) { + update_option( 'woocommerce_gzd_shipments_return_address_country', get_option( 'woocommerce_default_country' ) ); + } + } + } + } + + public static function product_options() { + global $post, $thepostid, $product_object; + + $_product = wc_get_product( $product_object ); + $shipments_product = wc_gzd_shipments_get_product( $_product ); + + $countries = WC()->countries->get_countries(); + $countries = array_merge( array( '0' => _x( 'Select a country', 'shipments', 'woocommerce-germanized' ) ), $countries ); + + woocommerce_wp_text_input( + array( + 'id' => '_hs_code', + 'label' => _x( 'HS-Code (Customs)', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => true, + 'description' => _x( 'The HS Code is a number assigned to every possible commodity that can be imported or exported from any country.', 'shipments', 'woocommerce-germanized' ), + 'value' => $shipments_product->get_hs_code( 'edit' ), + ) + ); + + woocommerce_wp_select( + array( + 'options' => $countries, + 'id' => '_manufacture_country', + 'label' => _x( 'Country of manufacture (Customs)', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => true, + 'description' => _x( 'The country of manufacture is needed for customs of international shipping.', 'shipments', 'woocommerce-germanized' ), + 'value' => $shipments_product->get_manufacture_country( 'edit' ), + ) + ); + + do_action( 'woocommerce_gzd_shipments_product_options', $shipments_product ); + } + + /** + * @param \WC_Product $product + */ + public static function save_product( $product ) { + $hs_code = isset( $_POST['_hs_code'] ) ? wc_clean( wp_unslash( $_POST['_hs_code'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Missing + $country = isset( $_POST['_manufacture_country'] ) ? wc_clean( wp_unslash( $_POST['_manufacture_country'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Missing + + $shipments_product = wc_gzd_shipments_get_product( $product ); + $shipments_product->set_hs_code( $hs_code ); + $shipments_product->set_manufacture_country( $country ); + + /** + * Remove legacy data upon saving in case it is not transmitted (e.g. DHL standalone plugin). + */ + if ( apply_filters( 'woocommerce_gzd_shipments_remove_legacy_customs_meta', isset( $_POST['_dhl_hs_code'] ) ? false : true, $product ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $product->delete_meta_data( '_dhl_hs_code' ); + $product->delete_meta_data( '_dhl_manufacture_country' ); + } + + do_action( 'woocommerce_gzd_shipments_save_product_options', $shipments_product ); + } + + public static function check_upload_dir() { + $dir = Package::get_upload_dir(); + $path = $dir['basedir']; + $dirname = basename( $path ); + + if ( @is_dir( $dir['basedir'] ) ) { // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged + return; + } + ?> +
+

wp-content/uploads/' . esc_html( $dirname ) . '' ); ?>

+
+ $value ) { + if ( isset( $value['id'] ) && $value['id'] === $id ) { + if ( ! empty( $type ) && $type !== $value['type'] ) { + continue; + } + return $key; + } + } + } + + return false; + } + + protected static function add_settings_after( $settings, $id, $insert = array(), $type = '' ) { + $key = self::get_setting_key_by_id( $settings, $id, $type ); + + if ( is_numeric( $key ) ) { + $key ++; + $settings = array_merge( array_merge( array_slice( $settings, 0, $key, true ), $insert ), array_slice( $settings, $key, count( $settings ) - 1, true ) ); + } else { + $settings += $insert; + } + + return $settings; + } + + public static function register_endpoint_settings( $settings, $current_section ) { + if ( '' === $current_section ) { + $endpoints = array( + array( + 'title' => _x( 'View Shipments', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Endpoint for the "My account → View shipments" page.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_view_shipments_endpoint', + 'type' => 'text', + 'default' => 'view-shipments', + 'desc_tip' => true, + ), + array( + 'title' => _x( 'View shipment', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Endpoint for the "My account → View shipment" page.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_view_shipment_endpoint', + 'type' => 'text', + 'default' => 'view-shipment', + 'desc_tip' => true, + ), + array( + 'title' => _x( 'Add Return Shipment', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Endpoint for the "My account → Add return shipment" page.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_add_return_shipment_endpoint', + 'type' => 'text', + 'default' => 'add-return-shipment', + 'desc_tip' => true, + ), + ); + + $settings = self::add_settings_after( $settings, 'woocommerce_myaccount_downloads_endpoint', $endpoints ); + } + + return $settings; + } + + public static function menu_return_count() { + global $submenu; + + if ( isset( $submenu['woocommerce'] ) ) { + + /** + * Filter to adjust whether to include requested return count in admin menu or not. + * + * @param boolean $show_count Whether to show count or not. + * + * @since 3.1.3 + * @package Vendidero/Germanized/Shipments + */ + if ( apply_filters( 'woocommerce_gzd_shipments_include_requested_return_count_in_menu', true ) && current_user_can( 'edit_others_shop_orders' ) ) { + $return_count = wc_gzd_get_shipment_count( 'requested', 'return' ); + + if ( $return_count ) { + foreach ( $submenu['woocommerce'] as $key => $menu_item ) { + if ( 0 === strpos( $menu_item[0], _x( 'Returns', 'shipments', 'woocommerce-germanized' ) ) ) { + $submenu['woocommerce'][ $key ][0] .= ' ' . number_format_i18n( $return_count ) . ''; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited + break; + } + } + } + } + } + } + + public static function get_admin_shipment_item_columns( $shipment ) { + $item_columns = array( + 'name' => array( + 'title' => _x( 'Item', 'shipments', 'woocommerce-germanized' ), + 'size' => 6, + 'order' => 5, + ), + 'quantity' => array( + 'title' => _x( 'Quantity', 'shipments', 'woocommerce-germanized' ), + 'size' => 3, + 'order' => 10, + ), + 'action' => array( + 'title' => _x( 'Actions', 'shipments', 'woocommerce-germanized' ), + 'size' => 3, + 'order' => 15, + ), + ); + + if ( 'return' === $shipment->get_type() ) { + $item_columns['return_reason'] = array( + 'title' => _x( 'Reason', 'shipments', 'woocommerce-germanized' ), + 'size' => 3, + 'order' => 7, + ); + + $item_columns['name']['size'] = 5; + $item_columns['quantity']['size'] = 2; + $item_columns['action']['size'] = 2; + } + + uasort( $item_columns, array( __CLASS__, 'sort_shipment_item_columns' ) ); + + /** + * Filter to adjust shipment item columns shown in admin view. + * + * @param array $item_columns The columns available. + * @param Shipment $shipment The shipment. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipments_meta_box_shipment_item_columns', $item_columns, $shipment ); + } + + protected static function sort_shipment_item_columns( $a, $b ) { + if ( $a['order'] === $b['order'] ) { + return 0; + } + + return ( $a['order'] < $b['order'] ) ? -1 : 1; + } + + public static function save_packaging_list() { + $current_key_list = array(); + $packaging_ids_after_save = array(); + + foreach ( wc_gzd_get_packaging_list() as $pack ) { + $current_key_list[] = $pack->get_id(); + } + + if ( isset( $_POST['packaging'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $packaging_post = wc_clean( wp_unslash( $_POST['packaging'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + $order = 0; + $available_types = array_keys( wc_gzd_get_packaging_types() ); + + foreach ( $packaging_post as $packaging ) { + $packaging = wc_clean( $packaging ); + $packaging_id = isset( $packaging['packaging_id'] ) ? absint( $packaging['packaging_id'] ) : 0; + $packaging_obj = wc_gzd_get_packaging( $packaging_id ); + + if ( $packaging_obj ) { + $packaging_obj->set_props( + array( + 'type' => ! in_array( $packaging['type'], $available_types, true ) ? 'cardboard' : $packaging['type'], + 'weight' => empty( $packaging['weight'] ) ? 0 : $packaging['weight'], + 'description' => empty( $packaging['description'] ) ? '' : $packaging['description'], + 'length' => empty( $packaging['length'] ) ? 0 : $packaging['length'], + 'width' => empty( $packaging['width'] ) ? 0 : $packaging['width'], + 'height' => empty( $packaging['height'] ) ? 0 : $packaging['height'], + 'max_content_weight' => empty( $packaging['max_content_weight'] ) ? 0 : $packaging['max_content_weight'], + 'order' => ++$order, + ) + ); + + if ( empty( $packaging_obj->get_description() ) ) { + if ( $packaging_obj->get_id() > 0 ) { + $packaging_obj->delete( true ); + continue; + } else { + continue; + } + } + + $packaging_obj->save(); + $packaging_ids_after_save[] = $packaging_obj->get_id(); + } + } + } + + $to_delete = array_diff( $current_key_list, $packaging_ids_after_save ); + + if ( ! empty( $to_delete ) ) { + foreach ( $to_delete as $delete_id ) { + if ( $packaging = wc_gzd_get_packaging( $delete_id ) ) { + $packaging->delete( true ); + } + } + } + } + + public static function save_return_reasons( $tab, $current_section ) { + if ( '' !== $current_section ) { + return; + } + + $reasons = array(); + + if ( isset( $_POST['shipment_return_reason'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $reasons_post = wc_clean( wp_unslash( $_POST['shipment_return_reason'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + $order = 0; + + foreach ( $reasons_post as $reason ) { + $code = isset( $reason['code'] ) ? $reason['code'] : ''; + $reason_text = isset( $reason['reason'] ) ? $reason['reason'] : ''; + + if ( empty( $code ) ) { + $code = sanitize_title( $reason_text ); + } + + if ( ! empty( $reason_text ) ) { + $reasons[] = array( + 'order' => ++$order, + 'code' => $code, + 'reason' => $reason_text, + ); + } + } + } + // phpcs:enable + + update_option( 'woocommerce_gzd_shipments_return_reasons', $reasons ); + } + + public static function output_return_reasons_field( $value ) { + ob_start(); + ?> + + + +
+ + + + + + + + + + + + + + '; + } + ?> + + + + + + +
 
+
+ + + + + + + + +
+ + + +
+ + + + + + + + + + + + + + + + + + + + + + + +
get_title() ); ?> get_status() ) ); ?> + get_date_start()->date_i18n( wc_date_format() ); + + printf( + '', + esc_attr( $report->get_date_start()->date( 'c' ) ), + esc_html( $report->get_date_start()->date_i18n( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ) ) ), + esc_html( $show_date ) + ); + ?> + + get_date_end()->date_i18n( wc_date_format() ); + + printf( + '', + esc_attr( $report->get_date_end()->date( 'c' ) ), + esc_html( $report->get_date_end()->date_i18n( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ) ) ), + esc_html( $show_date ) + ); + ?> + + get_total_weight(), wc_gzd_get_packaging_weight_unit() ) ); ?> + + get_total_count() ); ?> +
+ + + + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
 
+ + + + + + + + + + + + + + +
+
+ + + + 'shop_order', + 'bulk_action' => $report_action, + 'changed' => $changed, + 'ids' => join( ',', $ids ), + ); + + if ( Package::is_hpos_enabled() ) { + unset( $redirect_query_args['post_type'] ); + $redirect_query_args['page'] = 'wc-orders'; + } + + $redirect_to = add_query_arg( + $redirect_query_args, + $redirect_to + ); + + return esc_url_raw( $redirect_to ); + } else { + return $redirect_to; + } + } + + public static function define_order_bulk_actions( $actions ) { + $actions['gzd_create_shipments'] = _x( 'Create shipments', 'shipments', 'woocommerce-germanized' ); + + return $actions; + } + + public static function set_screen_option( $new_value, $option, $value ) { + + if ( in_array( $option, array( 'woocommerce_page_wc_gzd_shipments_per_page', 'woocommerce_page_wc_gzd_return_shipments_per_page' ), true ) ) { + return absint( $value ); + } + + return $new_value; + } + + public static function shipments_menu() { + add_submenu_page( 'woocommerce', _x( 'Shipments', 'shipments', 'woocommerce-germanized' ), _x( 'Shipments', 'shipments', 'woocommerce-germanized' ), 'edit_others_shop_orders', 'wc-gzd-shipments', array( __CLASS__, 'shipments_page' ) ); + add_submenu_page( 'woocommerce', _x( 'Returns', 'shipments', 'woocommerce-germanized' ), _x( 'Returns', 'shipments', 'woocommerce-germanized' ), 'edit_others_shop_orders', 'wc-gzd-return-shipments', array( __CLASS__, 'returns_page' ) ); + } + + /** + * @param Shipment $shipment + */ + public static function get_shipment_tracking_html( $shipment ) { + $tracking_html = ''; + + if ( $tracking_id = $shipment->get_tracking_id() ) { + + if ( $tracking_url = $shipment->get_tracking_url() ) { + $tracking_html = '' . $tracking_id . ''; + } else { + $tracking_html = '' . $tracking_id . ''; + } + } + + return $tracking_html; + } + + /** + * @param Table $table + */ + protected static function setup_table( $table ) { + global $wp_list_table; + + $wp_list_table = $table; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited + $doaction = $wp_list_table->current_action(); + + if ( $doaction ) { + check_admin_referer( 'bulk-shipments' ); + + $pagenum = $wp_list_table->get_pagenum(); + $parent_file = $wp_list_table->get_main_page(); + $sendback = remove_query_arg( array( 'deleted', 'ids', 'changed', 'bulk_action' ), wp_get_referer() ); + + if ( ! $sendback ) { + $sendback = admin_url( $parent_file ); + } + + $sendback = add_query_arg( 'paged', $pagenum, $sendback ); + $shipment_ids = array(); + + if ( isset( $_REQUEST['ids'] ) ) { + $shipment_ids = array_map( 'absint', explode( ',', wp_unslash( $_REQUEST['ids'] ) ) ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + } elseif ( ! empty( $_REQUEST['shipment'] ) ) { + $shipment_ids = array_map( 'absint', wp_unslash( $_REQUEST['shipment'] ) ); + } + + if ( ! empty( $shipment_ids ) ) { + $sendback = $wp_list_table->handle_bulk_actions( $doaction, $shipment_ids, $sendback ); + } + + $sendback = remove_query_arg( array( 'action', 'action2', '_status', 'bulk_edit', 'shipment' ), $sendback ); + + wp_safe_redirect( esc_url_raw( $sendback ) ); + exit(); + + } elseif ( ! empty( $_REQUEST['_wp_http_referer'] ) ) { + wp_safe_redirect( esc_url_raw( remove_query_arg( array( '_wp_http_referer', '_wpnonce' ), esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) ) ) ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotValidated + exit; + } + + $wp_list_table->set_bulk_notice(); + $wp_list_table->prepare_items(); + + add_screen_option( 'per_page' ); + } + + public static function setup_shipments_table() { + $table = new Table(); + + self::setup_table( $table ); + } + + public static function setup_returns_table() { + $table = new ReturnTable( array( 'type' => 'return' ) ); + + self::setup_table( $table ); + } + + public static function shipments_page() { + global $wp_list_table; + + ?> +
+

+
+ + output_notice(); + $_SERVER['REQUEST_URI'] = remove_query_arg( array( 'updated', 'changed', 'deleted', 'trashed', 'untrashed' ), ( isset( $_SERVER['REQUEST_URI'] ) ? esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) : admin_url( 'admin.php?page=wc-gzd-shipments' ) ) ); + ?> + + views(); ?> + +
+ + search_box( _x( 'Search shipments', 'shipments', 'woocommerce-germanized' ), 'shipment' ); ?> + + + + + + + + display(); ?> +
+ +
+
+
+ +
+

+
+ + output_notice(); + $_SERVER['REQUEST_URI'] = remove_query_arg( array( 'updated', 'changed', 'deleted', 'trashed', 'untrashed' ), ( isset( $_SERVER['REQUEST_URI'] ) ? esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) : admin_url( 'admin.php?page=wc-gzd-shipments' ) ) ); + ?> + + views(); ?> + +
+ + search_box( _x( 'Search returns', 'shipments', 'woocommerce-germanized' ), 'shipment' ); ?> + + + + + + + + display(); ?> +
+ +
+
+
+ id : ''; + $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; + + // Register admin styles. + wp_register_style( 'woocommerce_gzd_shipments_admin', Package::get_assets_url() . '/css/admin' . $suffix . '.css', array( 'woocommerce_admin_styles' ), Package::get_version() ); + + // Admin styles for WC pages only. + if ( in_array( $screen_id, self::get_screen_ids(), true ) ) { + wp_enqueue_style( 'woocommerce_gzd_shipments_admin' ); + } + + if ( 'woocommerce_page_wc-settings' === $screen_id && isset( $_GET['tab'] ) && in_array( $_GET['tab'], array( 'germanized-shipments', 'germanized-shipping_provider' ), true ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + wp_enqueue_style( 'woocommerce_gzd_shipments_admin' ); + } + + // Shipping zone methods + if ( 'woocommerce_page_wc-settings' === $screen_id && isset( $_GET['tab'] ) && 'shipping' === $_GET['tab'] && ( isset( $_GET['zone_id'] ) || isset( $_GET['instance_id'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + wp_enqueue_style( 'woocommerce_gzd_shipments_admin' ); + } + } + + public static function admin_scripts() { + global $post, $theorder; + + $screen = get_current_screen(); + $screen_id = $screen ? $screen->id : ''; + $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; + $post_id = isset( $post->ID ) ? $post->ID : ''; + $order_or_post_object = $post; + + if ( ( $theorder instanceof \WC_Order ) && self::is_order_meta_box_screen( $screen_id ) ) { + $order_or_post_object = $theorder; + } + + wp_register_script( 'wc-gzd-admin-shipment-label-backbone', Package::get_assets_url() . '/js/admin-shipment-label-backbone' . $suffix . '.js', array( 'jquery', 'woocommerce_admin', 'wc-backbone-modal' ), Package::get_version() ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.NotInFooter + wp_register_script( 'wc-gzd-admin-shipment', Package::get_assets_url() . '/js/admin-shipment' . $suffix . '.js', array( 'wc-gzd-admin-shipment-label-backbone' ), Package::get_version() ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.NotInFooter + wp_register_script( 'wc-gzd-admin-shipments', Package::get_assets_url() . '/js/admin-shipments' . $suffix . '.js', array( 'wc-admin-order-meta-boxes', 'wc-gzd-admin-shipment' ), Package::get_version() ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.NotInFooter + wp_register_script( 'wc-gzd-admin-shipments-table', Package::get_assets_url() . '/js/admin-shipments-table' . $suffix . '.js', array( 'woocommerce_admin', 'wc-gzd-admin-shipment-label-backbone' ), Package::get_version() ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.NotInFooter + wp_register_script( 'wc-gzd-admin-shipping-providers', Package::get_assets_url() . '/js/admin-shipping-providers' . $suffix . '.js', array( 'jquery', 'jquery-ui-sortable' ), Package::get_version() ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.NotInFooter + wp_register_script( 'wc-gzd-admin-shipping-provider-method', Package::get_assets_url() . '/js/admin-shipping-provider-method' . $suffix . '.js', array( 'jquery' ), Package::get_version() ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.NotInFooter + + // Orders. + if ( self::is_order_meta_box_screen( $screen_id ) ) { + wp_enqueue_script( 'wc-gzd-admin-shipments' ); + wp_enqueue_script( 'wc-gzd-admin-shipment' ); + + $order_order_post_id = $post_id; + + if ( self::is_order_meta_box_screen( $screen_id ) && isset( $order_or_post_object ) && is_callable( array( '\Automattic\WooCommerce\Utilities\OrderUtil', 'get_post_or_order_id' ) ) ) { + $order_order_post_id = \Automattic\WooCommerce\Utilities\OrderUtil::get_post_or_order_id( $order_or_post_object ); + } + + wp_localize_script( + 'wc-gzd-admin-shipments', + 'wc_gzd_admin_shipments_params', + array( + 'ajax_url' => admin_url( 'admin-ajax.php' ), + 'edit_shipments_nonce' => wp_create_nonce( 'edit-shipments' ), + 'order_id' => $order_order_post_id, + 'shipment_locked_excluded_fields' => array( 'status' ), + 'i18n_remove_shipment_notice' => _x( 'Do you really want to delete the shipment?', 'shipments', 'woocommerce-germanized' ), + 'remove_label_nonce' => wp_create_nonce( 'remove-shipment-label' ), + 'edit_label_nonce' => wp_create_nonce( 'edit-shipment-label' ), + 'send_return_notification_nonce' => wp_create_nonce( 'send-return-shipment-notification' ), + 'refresh_packaging_nonce' => wp_create_nonce( 'refresh-shipment-packaging' ), + 'confirm_return_request_nonce' => wp_create_nonce( 'confirm-return-request' ), + 'i18n_remove_label_notice' => _x( 'Do you really want to delete the label?', 'shipments', 'woocommerce-germanized' ), + 'i18n_create_label_enabled' => _x( 'Create new label', 'shipments', 'woocommerce-germanized' ), + 'i18n_create_label_disabled' => _x( 'Please save the shipment before creating a new label', 'shipments', 'woocommerce-germanized' ), + ) + ); + } + + // Table + if ( 'woocommerce_page_wc-gzd-shipments' === $screen_id || 'woocommerce_page_wc-gzd-return-shipments' === $screen_id ) { + wp_enqueue_script( 'wc-gzd-admin-shipments-table' ); + + $bulk_actions = array(); + + foreach ( self::get_bulk_action_handlers() as $handler ) { + $bulk_actions[ sanitize_key( $handler->get_action() ) ] = array( + 'title' => $handler->get_title(), + 'nonce' => wp_create_nonce( $handler->get_nonce_name() ), + ); + } + + wp_localize_script( + 'wc-gzd-admin-shipments-table', + 'wc_gzd_admin_shipments_table_params', + array( + 'ajax_url' => admin_url( 'admin-ajax.php' ), + 'search_orders_nonce' => wp_create_nonce( 'search-orders' ), + 'search_shipping_provider_nonce' => wp_create_nonce( 'search-shipping-provider' ), + 'bulk_actions' => $bulk_actions, + ) + ); + } + + wp_localize_script( + 'wc-gzd-admin-shipment-label-backbone', + 'wc_gzd_admin_shipment_label_backbone_params', + array( + 'ajax_url' => admin_url( 'admin-ajax.php' ), + 'i18n_modal_close' => _x( 'Close', 'shipments-close-modal', 'woocommerce-germanized' ), + 'create_label_form_nonce' => wp_create_nonce( 'create-shipment-label-form' ), + 'create_label_nonce' => wp_create_nonce( 'create-shipment-label' ), + ) + ); + + // Shipping provider settings + if ( 'woocommerce_page_wc-settings' === $screen_id && isset( $_GET['tab'] ) && 'germanized-shipping_provider' === $_GET['tab'] && empty( $_GET['provider'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + wp_enqueue_script( 'wc-gzd-admin-shipping-providers' ); + + wp_localize_script( + 'wc-gzd-admin-shipping-providers', + 'wc_gzd_admin_shipping_providers_params', + array( + 'ajax_url' => admin_url( 'admin-ajax.php' ), + 'edit_shipping_providers_nonce' => wp_create_nonce( 'edit-shipping-providers' ), + 'remove_shipping_provider_nonce' => wp_create_nonce( 'remove-shipping-provider' ), + 'sort_shipping_provider_nonce' => wp_create_nonce( 'sort-shipping-provider' ), + 'i18n_remove_shipping_provider_notice' => _x( 'Do you really want to delete the shipping provider? Some of your existing shipments might be linked to that provider and might need adjustments.', 'shipments', 'woocommerce-germanized' ), + ) + ); + } + + // Shipping provider method + if ( 'woocommerce_page_wc-settings' === $screen_id && isset( $_GET['tab'] ) && 'shipping' === $_GET['tab'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + /** + * Older third-party shipping methods may not support instance-settings and will have their settings + * output in a separate section under Settings > Shipping. + */ + if ( ( isset( $_GET['zone_id'] ) || isset( $_GET['instance_id'] ) ) || ( isset( $_GET['section'] ) && 'classes' !== $_GET['section'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + wp_enqueue_script( 'wc-gzd-admin-shipping-provider-method' ); + $providers = array_filter( array_keys( wc_gzd_get_shipping_provider_select() ) ); + + wp_localize_script( + 'wc-gzd-admin-shipping-provider-method', + 'wc_gzd_admin_shipping_provider_method_params', + array( + 'shipping_providers' => $providers, + ) + ); + } + } + } + + /** + * @return BulkActionHandler[] $handler + */ + public static function get_bulk_action_handlers() { + if ( is_null( self::$bulk_handlers ) ) { + self::$bulk_handlers = array(); + + /** + * Filter to register new BulkActionHandler for certain Shipment bulk actions. + * + * @param array $handlers Array containing key => classname. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $handlers = apply_filters( + 'woocommerce_gzd_shipments_table_bulk_action_handlers', + array( + 'labels' => '\Vendidero\Germanized\Shipments\Admin\BulkLabel', + ) + ); + + foreach ( $handlers as $key => $handler ) { + self::$bulk_handlers[ $key ] = new $handler(); + } + } + + return self::$bulk_handlers; + } + + public static function get_bulk_action_handler( $action ) { + $handlers = self::get_bulk_action_handlers(); + + return array_key_exists( $action, $handlers ) ? $handlers[ $action ] : false; + } + + /** + * Helper function to determine whether the current screen is an order edit screen. + * + * @param string $screen_id Screen ID. + * + * @return bool Whether the current screen is an order edit screen. + */ + protected static function is_order_meta_box_screen( $screen_id ) { + return in_array( str_replace( 'edit-', '', $screen_id ), self::get_order_screen_ids(), true ); + } + + public static function get_order_screen_id() { + return function_exists( 'wc_get_page_screen_id' ) ? wc_get_page_screen_id( 'shop-order' ) : 'shop_order'; + } + + protected static function get_order_screen_ids() { + $screen_ids = array(); + + foreach ( wc_get_order_types() as $type ) { + $screen_ids[] = $type; + $screen_ids[] = 'edit-' . $type; + } + + $screen_ids[] = self::get_order_screen_id(); + + return array_filter( $screen_ids ); + } + + public static function get_screen_ids() { + $screen_ids = array( + 'woocommerce_page_wc-gzd-shipments', + 'woocommerce_page_wc-gzd-return-shipments', + ); + + return array_merge( $screen_ids, self::get_order_screen_ids() ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/BulkActionHandler.php b/packages/woocommerce-germanized-shipments/src/Admin/BulkActionHandler.php new file mode 100644 index 000000000..805ed7d7b --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/BulkActionHandler.php @@ -0,0 +1,163 @@ +notices = array_filter( (array) get_user_meta( get_current_user_id(), $this->get_notice_option_name(), true ) ); + } + + protected function get_notice_option_name() { + $action = sanitize_key( $this->get_action() ); + + return "woocommerce_gzd_shipments_{$action}_bulk_notices"; + } + + abstract public function get_title(); + + public function get_nonce_name() { + $action = sanitize_key( $this->get_action() ); + + return "woocommerce_gzd_shipments_{$action}"; + } + + public function get_shipment_type() { + return $this->type; + } + + public function set_shipment_type( $type ) { + $this->type = $type; + } + + public function get_success_redirect_url() { + $page = 'wc-gzd-shipments'; + + if ( 'simple' !== $this->get_shipment_type() ) { + $page = 'wc-gzd-' . $this->get_shipment_type() . '-shipments'; + } + + return admin_url( 'admin.php?page=' . $page . '&bulk_action_handling=finished¤t_bulk_action=' . sanitize_key( $this->get_action() ) ); + } + + public function get_step() { + return $this->step; + } + + public function set_step( $step ) { + $this->step = $step; + } + + public function get_notices( $type = 'error' ) { + $notices = array_key_exists( $type, $this->notices ) ? $this->notices[ $type ] : array(); + + return $notices; + } + + public function get_success_message() { + return _x( 'Successfully processed shipments.', 'shipments', 'woocommerce-germanized' ); + } + + public function admin_handled() { + + } + + public function admin_after_error() { + + } + + public function add_notice( $notice, $type = 'error' ) { + if ( ! isset( $this->notices[ $type ] ) ) { + $this->notices[ $type ] = array(); + } + + $this->notices[ $type ][] = $notice; + } + + public function update_notices() { + update_user_meta( get_current_user_id(), $this->get_notice_option_name(), $this->notices ); + } + + public function reset( $is_new = false ) { + delete_user_meta( get_current_user_id(), $this->get_notice_option_name() ); + } + + abstract public function get_action(); + + public function get_max_step() { + return (int) ceil( count( $this->get_ids() ) / $this->get_limit() ); + } + + abstract public function get_limit(); + + public function get_total() { + return count( $this->get_ids() ); + } + + abstract public function handle(); + + public function set_ids( $ids ) { + $this->ids = $ids; + } + + public function get_ids() { + return $this->ids; + } + + public function get_current_ids() { + return array_slice( $this->get_ids(), ( $this->get_step() - 1 ) * $this->get_limit(), $this->get_limit() ); + } + + /** + * Get count of records exported. + * + * @since 3.0.6 + * @return int + */ + public function get_total_processed() { + return ( $this->get_step() * $this->get_limit() ); + } + + /** + * Get total % complete. + * + * @since 3.0.6 + * @return int + */ + public function get_percent_complete() { + return floor( ( $this->get_total_processed() / $this->get_total() ) * 100 ); + } + + public function is_last_step() { + $current_step = $this->get_step(); + $max_step = $this->get_max_step(); + + if ( $max_step === $current_step ) { + return true; + } + + return false; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/BulkLabel.php b/packages/woocommerce-germanized-shipments/src/Admin/BulkLabel.php new file mode 100644 index 000000000..2fada1f79 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/BulkLabel.php @@ -0,0 +1,215 @@ +get_file_option_name(), true ); + + if ( $file ) { + $uploads = Package::get_upload_dir(); + $path = trailingslashit( $uploads['basedir'] ) . $file; + + return $path; + } + + return ''; + } + + protected function update_file( $path ) { + update_user_meta( get_current_user_id(), $this->get_file_option_name(), $path ); + } + + protected function get_file_option_name() { + $action = sanitize_key( $this->get_action() ); + + return "woocommerce_gzd_shipments_{$action}_bulk_path"; + } + + public function get_filename() { + if ( $file = $this->get_file() ) { + return basename( $file ); + } + + return ''; + } + + public function reset( $is_new = false ) { + parent::reset( $is_new ); + + if ( $is_new ) { + delete_user_meta( get_current_user_id(), $this->get_file_option_name() ); + delete_user_meta( get_current_user_id(), $this->get_files_option_name() ); + } + } + + protected function get_download_button() { + $download_button = ''; + + if ( ( $path = $this->get_file() ) && file_exists( $path ) ) { + + $download_url = add_query_arg( + array( + 'action' => 'wc-gzd-download-export-shipment-label', + 'force' => 'no', + ), + wp_nonce_url( admin_url(), 'download-export-shipment-label' ) + ); + + $download_button = '' . esc_html_x( 'Download labels', 'shipments', 'woocommerce-germanized' ) . ''; + } + + return $download_button; + } + + public function get_success_message() { + $download_button = $this->get_download_button(); + + if ( empty( $download_button ) ) { + return sprintf( _x( 'The chosen shipments were not suitable for automatic label creation. Please check the shipping provider option of the corresponding shipments.', 'shipments', 'woocommerce-germanized' ), $download_button ); + } else { + return sprintf( _x( 'Successfully generated labels. %s', 'shipments', 'woocommerce-germanized' ), $download_button ); + } + } + + public function admin_after_error() { + $download_button = $this->get_download_button(); + + if ( ! empty( $download_button ) ) { + echo '

' . sprintf( esc_html_x( 'Labels partially generated. %s', 'shipments', 'woocommerce-germanized' ), $download_button ) . '

'; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + } + } + + protected function get_files_option_name() { + $action = sanitize_key( $this->get_action() ); + + return "woocommerce_gzd_shipments_{$action}_bulk_files"; + } + + protected function get_files() { + $files = get_user_meta( get_current_user_id(), $this->get_files_option_name(), true ); + + if ( empty( $files ) || ! is_array( $files ) ) { + $files = array(); + } + + return $files; + } + + protected function add_file( $path ) { + $files = $this->get_files(); + + if ( ! in_array( $path, $files, true ) ) { + $files[] = $path; + update_user_meta( get_current_user_id(), $this->get_files_option_name(), $files ); + } + } + + public function handle() { + $current = $this->get_current_ids(); + + if ( ! empty( $current ) ) { + foreach ( $current as $shipment_id ) { + $label = false; + + if ( $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + if ( $shipment->supports_label() ) { + if ( $shipment->needs_label() ) { + $result = $shipment->create_label(); + + if ( is_wp_error( $result ) ) { + $result = wc_gzd_get_shipment_error( $result ); + } + + if ( is_wp_error( $result ) ) { + foreach ( $result->get_error_messages_by_type() as $type => $messages ) { + foreach ( $messages as $message ) { + if ( 'soft' === $type ) { + $this->add_notice( sprintf( _x( 'Notice while creating label for %1$s: %2$s', 'shipments', 'woocommerce-germanized' ), '' . sprintf( _x( 'shipment #%d', 'shipments', 'woocommerce-germanized' ), $shipment_id ) . '', $message ), 'info' ); + } else { + $this->add_notice( sprintf( _x( 'Error while creating label for %1$s: %2$s', 'shipments', 'woocommerce-germanized' ), '' . sprintf( _x( 'shipment #%d', 'shipments', 'woocommerce-germanized' ), $shipment_id ) . '', $message ), 'error' ); + } + } + } + } + + if ( $shipment->has_label() ) { + $label = $shipment->get_label(); + } + } else { + $label = $shipment->get_label(); + } + } + } + + if ( $label ) { + $this->add_file( $label->get_file() ); + } + } + } + + if ( $this->is_last_step() ) { + try { + $files = $this->get_files(); + $pdf = new PDFMerger(); + + if ( ! empty( $files ) ) { + foreach ( $files as $file ) { + if ( ! file_exists( $file ) ) { + continue; + } + + $pdf->add( $file ); + } + + /** + * Filter to adjust the default filename chosen for bulk exporting shipment labels. + * + * @param string $filename The filename. + * @param BulkLabel $this The `BulkLabel instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/shipments + */ + $filename = apply_filters( 'woocommerce_gzd_shipment_labels_bulk_filename', 'export.pdf', $this ); + $file = $pdf->output( $filename, 'S' ); + + if ( $path = wc_gzd_shipments_upload_data( $filename, $file ) ) { + $this->update_file( $path ); + } + } + } catch ( Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } + } + + $this->update_notices(); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/MetaBox.php b/packages/woocommerce-germanized-shipments/src/Admin/MetaBox.php new file mode 100644 index 000000000..d34957581 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/MetaBox.php @@ -0,0 +1,178 @@ +get_shipments() as $shipment ) { + $id = $shipment->get_id(); + $props = array(); + + // Update items + self::refresh_shipment_items( $order, $shipment ); + + // Do only update props if they exist + if ( isset( $_POST['shipment_weight'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['weight'] = wc_clean( wp_unslash( $_POST['shipment_weight'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_length'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['length'] = wc_clean( wp_unslash( $_POST['shipment_length'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_width'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['width'] = wc_clean( wp_unslash( $_POST['shipment_width'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_height'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['height'] = wc_clean( wp_unslash( $_POST['shipment_height'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_shipping_method'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['shipping_method'] = wc_clean( wp_unslash( $_POST['shipment_shipping_method'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_tracking_id'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['tracking_id'] = wc_clean( wp_unslash( $_POST['shipment_tracking_id'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_packaging_id'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['packaging_id'] = wc_clean( wp_unslash( $_POST['shipment_packaging_id'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_shipping_provider'][ $id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $provider = wc_clean( wp_unslash( $_POST['shipment_shipping_provider'][ $id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + $providers = wc_gzd_get_shipping_providers(); + + if ( empty( $provider ) || array_key_exists( $provider, $providers ) ) { + $props['shipping_provider'] = $provider; + } + } + + $new_status = isset( $_POST['shipment_status'][ $id ] ) ? str_replace( 'gzd-', '', wc_clean( wp_unslash( $_POST['shipment_status'][ $id ] ) ) ) : 'draft'; // phpcs:ignore WordPress.Security.NonceVerification.Missing + + // Sync the shipment - make sure gets refresh on status switch (e.g. from shipped to processing) + if ( $shipment->is_editable() || in_array( $new_status, wc_gzd_get_shipment_editable_statuses(), true ) ) { + $shipment->sync( $props ); + } + } + } + + /** + * @param Order $order + * @param bool $shipment + */ + public static function refresh_shipment_items( &$order, &$shipment = false ) { + $shipments = $shipment ? array( $shipment ) : $order->get_shipments(); + + foreach ( $shipments as $shipment ) { + $id = $shipment->get_id(); + + if ( ! $shipment->is_editable() ) { + continue; + } + + // Update items + foreach ( $shipment->get_items() as $item ) { + $item_id = $item->get_id(); + $props = array(); + + // Set quantity to 1 by default + if ( $shipment->is_editable() ) { + $props['quantity'] = 1; + } + + if ( isset( $_POST['shipment_item'][ $id ]['quantity'][ $item_id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['quantity'] = absint( wp_unslash( $_POST['shipment_item'][ $id ]['quantity'][ $item_id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['shipment_item'][ $id ]['return_reason_code'][ $item_id ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $props['return_reason_code'] = wc_clean( wp_unslash( $_POST['shipment_item'][ $id ]['return_reason_code'][ $item_id ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + $item->sync( $props ); + } + } + } + + /** + * @param Order $order + */ + public static function refresh_status( &$order ) { + foreach ( $order->get_shipments() as $shipment ) { + $id = $shipment->get_id(); + $status = isset( $_POST['shipment_status'][ $id ] ) ? wc_clean( wp_unslash( $_POST['shipment_status'][ $id ] ) ) : 'draft'; // phpcs:ignore WordPress.Security.NonceVerification.Missing + + if ( ! wc_gzd_is_shipment_status( $status ) ) { + $status = 'draft'; + } + + $shipment->set_status( $status ); + } + } + + protected static function init_order_object( $post ) { + if ( is_callable( array( '\Automattic\WooCommerce\Utilities\OrderUtil', 'init_theorder_object' ) ) ) { + \Automattic\WooCommerce\Utilities\OrderUtil::init_theorder_object( $post ); + } else { + global $post, $thepostid, $theorder; + + if ( ! is_int( $thepostid ) ) { + $thepostid = $post->ID; + } + + if ( ! is_object( $theorder ) ) { + $theorder = wc_get_order( $thepostid ); + } + } + } + + /** + * Output the metabox. + * + * @param \WP_Post $post + */ + public static function output( $post ) { + global $theorder; + + self::init_order_object( $post ); + + $order = $theorder; + $order_shipment = wc_gzd_get_shipment_order( $order ); + $active_shipment = isset( $_GET['shipment_id'] ) ? absint( $_GET['shipment_id'] ) : 0; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + include Package::get_path() . '/includes/admin/views/html-order-shipments.php'; + } + + /** + * Save meta box data. + * + * @param int $post_id + */ + public static function save( $order_id ) { + // Get order object. + $order_shipment = wc_gzd_get_shipment_order( $order_id ); + + self::refresh_shipments( $order_shipment ); + + $order_shipment->validate_shipments( array( 'save' => false ) ); + + // Refresh status just before saving + self::refresh_status( $order_shipment ); + + $order_shipment->save(); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/ProviderSettings.php b/packages/woocommerce-germanized-shipments/src/Admin/ProviderSettings.php new file mode 100644 index 000000000..56218e2a0 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/ProviderSettings.php @@ -0,0 +1,260 @@ +get_shipping_providers(); + + if ( ! empty( $provider_name ) && 'new' !== $provider_name ) { + $provider = $helper->get_shipping_provider( $provider_name ); + } else { + $provider = new Simple(); + } + } + + return $provider; + } + + public static function get_help_link() { + if ( $provider = self::get_current_provider() ) { + return $provider->get_help_link(); + } else { + return 'https://vendidero.de/dokument/versanddienstleister-verwalten'; + } + } + + public static function get_next_pointers_link( $provider_name = false ) { + $providers = wc_gzd_get_shipping_providers(); + $next_url = admin_url( 'admin.php?page=wc-settings&tab=germanized-emails&tutorial=yes' ); + $provider_indexes = array(); + $provider_counts = array(); + $count = 0; + + foreach ( $providers as $provider_key => $provider ) { + if ( is_a( $provider, '\Vendidero\Germanized\Shipments\ShippingProvider\Auto' ) && ! empty( $provider->get_settings_help_pointers() ) ) { + $provider_indexes[ $provider_key ] = $count; + $provider_counts[ $count ] = $provider_key; + $count++; + } + } + + $next_index = isset( $provider_indexes[ $provider_name ] ) ? $provider_indexes[ $provider_name ] + 1 : -1; + + // By default use the first provider + if ( ! $provider_name ) { + $next_index = 0; + } + + if ( isset( $provider_counts[ $next_index ] ) ) { + $next_provider = $providers[ $provider_counts[ $next_index ] ]; + $next_url = add_query_arg( array( 'tutorial' => 'yes' ), $next_provider->get_edit_link() ); + } + + return $next_url; + } + + public static function get_pointers( $section ) { + $pointers = array(); + + if ( $provider = self::get_current_provider() ) { + if ( is_a( $provider, '\Vendidero\Germanized\Shipments\ShippingProvider\Auto' ) ) { + $pointers = $provider->get_settings_help_pointers( $section ); + } + } else { + $pointers = array( + 'pointers' => array( + 'provider' => array( + 'target' => '.wc-gzd-setting-tab-rows tr:first-child .wc-gzd-shipping-provider-title a.wc-gzd-shipping-provider-edit-link', + 'next' => 'activate', + 'next_url' => '', + 'next_trigger' => array(), + 'options' => array( + 'content' => '

' . esc_html_x( 'Shipping Provider', 'shipments', 'woocommerce-germanized' ) . '

' . esc_html_x( 'You may find all the available shipping providers as a list here. Click on the link to edit the provider-specific settings.', 'shipments', 'woocommerce-germanized' ) . '

', + 'position' => array( + 'edge' => 'left', + 'align' => 'left', + ), + ), + ), + 'activate' => array( + 'target' => '.wc-gzd-setting-tab-rows tr:first-child .wc-gzd-shipping-provider-activated .woocommerce-gzd-input-toggle-trigger', + 'next' => 'new', + 'next_url' => '', + 'next_trigger' => array(), + 'options' => array( + 'content' => '

' . esc_html_x( 'Activate', 'shipments', 'woocommerce-germanized' ) . '

' . esc_html_x( 'Activate or deactivate a shipping provider by toggling this button.', 'shipments', 'woocommerce-germanized' ) . '

', + 'position' => array( + 'edge' => 'right', + 'align' => 'left', + ), + ), + ), + 'new' => array( + 'target' => 'ul.wc-gzd-settings-breadcrumb .breadcrumb-item-active a.page-title-action:first', + 'next' => '', + 'next_url' => self::get_next_pointers_link(), + 'next_trigger' => array(), + 'options' => array( + 'content' => '

' . esc_html_x( 'Add new', 'shipments', 'woocommerce-germanized' ) . '

' . esc_html_x( 'You may want to manually add a new shipping provider in case an automatic integration does not exist.', 'shipments', 'woocommerce-germanized' ) . '

', + 'position' => array( + 'edge' => 'top', + 'align' => 'top', + ), + ), + ), + ), + ); + } + + return $pointers; + } + + public static function get_description() { + if ( $provider = self::get_current_provider() ) { + return $provider->get_description( 'edit' ); + } + + return ''; + } + + public static function get_breadcrumb( $current_section = '' ) { + $provider = self::get_current_provider(); + + $breadcrumb[] = array( + 'class' => 'tab', + 'href' => $provider ? admin_url( 'admin.php?page=wc-settings&tab=germanized-shipping_provider' ) : '', + 'title' => ! $provider ? self::get_breadcrumb_label( _x( 'Shipping Provider', 'shipments', 'woocommerce-germanized' ) ) : _x( 'Shipping Provider', 'shipments', 'woocommerce-germanized' ), + ); + + if ( $provider = self::get_current_provider() ) { + $breadcrumb[] = array( + 'class' => 'section', + 'href' => ! empty( $current_section ) ? $provider->get_edit_link() : '', + 'title' => ( $provider->get_id() <= 0 && '' === $provider->get_title() ) ? self::get_breadcrumb_label( _x( 'New', 'shipments-shipping-provider', 'woocommerce-germanized' ), $current_section ) : self::get_breadcrumb_label( $provider->get_title(), $current_section ), + ); + } + + if ( ! empty( $current_section ) ) { + $breadcrumb[] = array( + 'class' => 'section', + 'href' => '', + 'title' => self::get_section_title( $current_section ), + ); + } + + return $breadcrumb; + } + + protected static function get_section_title( $section = '' ) { + $sections = self::get_sections(); + $section_label = isset( $sections[ $section ] ) ? $sections[ $section ] : ''; + + return $section_label; + } + + protected static function get_breadcrumb_label( $label, $current_section = '' ) { + $help_link = self::get_help_link(); + $provider = self::get_current_provider(); + + if ( $provider && empty( $current_section ) ) { + if ( ! empty( $help_link ) ) { + $label = $label . '' . _x( 'Learn more', 'shipments', 'woocommerce-germanized' ) . ''; + } + + if ( ! empty( $provider->get_signup_link() ) ) { + $label = $label . '' . _x( 'Not yet a customer?', 'shipments', 'woocommerce-germanized' ) . ''; + } + } elseif ( ! $provider ) { + $label = $label . '' . _x( 'Add provider', 'shipments', 'woocommerce-germanized' ) . ''; + + if ( ! empty( $help_link ) ) { + $label = $label . '' . _x( 'Learn more', 'shipments', 'woocommerce-germanized' ) . ''; + } + } + + return $label; + } + + public static function save( $section = '' ) { + if ( $provider = self::get_current_provider() ) { + $is_new = $provider->get_id() <= 0 ? true : false; + + $provider->update_settings( $section, null, false ); + + if ( $is_new ) { + if ( empty( $provider->get_tracking_desc_placeholder( 'edit' ) ) ) { + $provider->set_tracking_desc_placeholder( $provider->get_default_tracking_desc_placeholder() ); + } + + if ( empty( $provider->get_tracking_url_placeholder( 'edit' ) ) ) { + $provider->set_tracking_url_placeholder( $provider->get_default_tracking_url_placeholder() ); + } + } + + if ( isset( $_GET['provider'] ) && 'new' === $_GET['provider'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + add_filter( 'woocommerce_gzd_shipments_shipping_provider_is_manual_creation_request', '__return_true', 15 ); + } + + $provider->save(); + + if ( isset( $_GET['provider'] ) && 'new' === $_GET['provider'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + remove_filter( 'woocommerce_gzd_shipments_shipping_provider_is_manual_creation_request', '__return_true', 15 ); + } + + if ( $is_new ) { + $url = admin_url( 'admin.php?page=wc-settings&tab=germanized-shipping_provider&provider=' . $provider->get_name() ); + wp_safe_redirect( $url ); + } + } + } + + public static function get_settings( $current_section = '' ) { + if ( $provider = self::get_current_provider() ) { + return $provider->get_settings( $current_section ); + } else { + return array(); + } + } + + public static function output_providers() { + global $hide_save_button; + + $hide_save_button = true; + self::provider_screen(); + } + + protected static function provider_screen() { + $helper = Helper::instance(); + $providers = $helper->get_shipping_providers(); + $providers = apply_filters( 'woocommerce_gzd_shipment_admin_provider_list', $providers ); + + include_once Package::get_path() . '/includes/admin/views/html-settings-provider-list.php'; + } + + public static function get_sections() { + if ( $provider = self::get_current_provider() ) { + return $provider->get_setting_sections(); + } else { + return array(); + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/ReturnTable.php b/packages/woocommerce-germanized-shipments/src/Admin/ReturnTable.php new file mode 100644 index 000000000..be9ffee3b --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/ReturnTable.php @@ -0,0 +1,101 @@ +'; + $columns['title'] = _x( 'Title', 'shipments', 'woocommerce-germanized' ); + $columns['date'] = _x( 'Date', 'shipments', 'woocommerce-germanized' ); + $columns['status'] = _x( 'Status', 'shipments', 'woocommerce-germanized' ); + $columns['items'] = _x( 'Items', 'shipments', 'woocommerce-germanized' ); + $columns['sender'] = _x( 'Sender', 'shipments', 'woocommerce-germanized' ); + $columns['weight'] = _x( 'Weight', 'shipments', 'woocommerce-germanized' ); + $columns['dimensions'] = _x( 'Dimensions', 'shipments', 'woocommerce-germanized' ); + $columns['order'] = _x( 'Order', 'shipments', 'woocommerce-germanized' ); + $columns['actions'] = _x( 'Actions', 'shipments', 'woocommerce-germanized' ); + + return $columns; + } + + /** + * @param ReturnShipment $shipment + * @param $actions + * + * @return mixed + */ + protected function get_custom_actions( $shipment, $actions ) { + + if ( isset( $actions['shipped'] ) ) { + unset( $actions['shipped'] ); + } + + if ( ! $shipment->has_status( 'delivered' ) && ! $shipment->has_status( 'requested' ) ) { + $actions['received'] = array( + 'url' => wp_nonce_url( admin_url( 'admin-ajax.php?action=woocommerce_gzd_update_shipment_status&status=delivered&shipment_id=' . $shipment->get_id() ), 'update-shipment-status' ), + 'name' => _x( 'Delivered', 'shipments', 'woocommerce-germanized' ), + 'action' => 'delivered', + ); + } + + if ( $shipment->has_status( 'processing' ) ) { + $actions['email_notification'] = array( + 'url' => wp_nonce_url( admin_url( 'admin-ajax.php?action=woocommerce_gzd_send_return_shipment_notification_email&shipment_id=' . $shipment->get_id() ), 'send-return-shipment-notification' ), + 'name' => _x( 'Send notification to customer', 'shipments', 'woocommerce-germanized' ), + 'action' => 'send-return-notification email', + ); + } + + if ( $shipment->is_customer_requested() && $shipment->has_status( 'requested' ) ) { + $actions['confirm'] = array( + 'url' => wp_nonce_url( admin_url( 'admin-ajax.php?action=woocommerce_gzd_confirm_return_request&shipment_id=' . $shipment->get_id() ), 'confirm-return-request' ), + 'name' => _x( 'Confirm return request', 'shipments', 'woocommerce-germanized' ), + 'action' => 'confirm', + ); + } + + return $actions; + } + + public function get_main_page() { + return 'admin.php?page=wc-gzd-return-shipments'; + } + + protected function get_custom_bulk_actions( $actions ) { + $actions['confirm_requests'] = _x( 'Confirm open return requests', 'shipments', 'woocommerce-germanized' ); + + return $actions; + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param ReturnShipment $shipment The current shipment object. + */ + public function column_sender( $shipment ) { + $address = $shipment->get_formatted_sender_address(); + + if ( $address ) { + echo '' . esc_html( preg_replace( '##i', ', ', $address ) ) . ''; + } else { + echo '–'; + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/Settings.php b/packages/woocommerce-germanized-shipments/src/Admin/Settings.php new file mode 100644 index 000000000..c6cfb8b56 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/Settings.php @@ -0,0 +1,612 @@ + array( + 'menu' => array( + 'target' => '.wc-gzd-settings-breadcrumb .page-title-action:last', + 'next' => 'default', + 'next_url' => '', + 'next_trigger' => array(), + 'options' => array( + 'content' => '

' . esc_html_x( 'Manage shipments', 'shipments', 'woocommerce-germanized' ) . '

' . esc_html_x( 'To view all your existing shipments in a list you might follow this link or click on the shipments link within the WooCommerce sub-menu.', 'shipments', 'woocommerce-germanized' ) . '

', + 'position' => array( + 'edge' => 'left', + 'align' => 'left', + ), + ), + ), + 'default' => array( + 'target' => '#woocommerce_gzd_shipments_notify_enable-toggle', + 'next' => 'auto', + 'next_url' => '', + 'next_trigger' => array(), + 'options' => array( + 'content' => '

' . esc_html_x( 'E-Mail Notification', 'shipments', 'woocommerce-germanized' ) . '

' . esc_html_x( 'By enabling this option customers receive an email notification as soon as a shipment is marked as shipped.', 'shipments', 'woocommerce-germanized' ) . '

', + 'position' => array( + 'edge' => 'left', + 'align' => 'left', + ), + ), + ), + 'auto' => array( + 'target' => '#woocommerce_gzd_shipments_auto_enable-toggle', + 'next' => 'returns', + 'next_url' => '', + 'next_trigger' => array(), + 'options' => array( + 'content' => '

' . esc_html_x( 'Automation', 'shipments', 'woocommerce-germanized' ) . '

' . esc_html_x( 'Decide whether you want to automatically create shipments to orders reaching a specific status. You can always adjust your shipments by manually editing the shipment within the edit order screen.', 'shipments', 'woocommerce-germanized' ) . '

', + 'position' => array( + 'edge' => 'left', + 'align' => 'left', + ), + ), + ), + 'returns' => array( + 'target' => '#shipments_return_options-description', + 'next' => '', + 'next_url' => $next_url, + 'next_trigger' => array(), + 'options' => array( + 'content' => '

' . esc_html_x( 'Returns', 'shipments', 'woocommerce-germanized' ) . '

' . sprintf( _x( 'Germanized can help you to minimize manual work while handling customer returns. Learn more about returns within our %s.', 'shipments', 'woocommerce-germanized' ), '' . _x( 'documentation', 'shipments', 'woocommerce-germanized' ) . '' ) . '

', + 'position' => array( + 'edge' => 'top', + 'align' => 'top', + ), + ), + ), + ), + ); + } + + return $pointers; + } + + protected static function get_general_settings() { + + $statuses = array_diff_key( wc_gzd_get_shipment_statuses(), array_flip( array( 'gzd-requested' ) ) ); + + $settings = array( + array( + 'title' => '', + 'type' => 'title', + 'id' => 'shipments_options', + ), + + array( + 'title' => _x( 'Notify', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Notify customers about new shipments.', 'shipments', 'woocommerce-germanized' ) . '
' . sprintf( _x( 'Notify customers by email as soon as a shipment is marked as shipped. %s the notification email.', 'shipments', 'woocommerce-germanized' ), '' . _x( 'Manage', 'shipments notification', 'woocommerce-germanized' ) . '' ) . '
', + 'id' => 'woocommerce_gzd_shipments_notify_enable', + 'default' => 'yes', + 'type' => 'gzd_toggle', + ), + + array( + 'title' => _x( 'Default provider', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Select a default shipping provider which will be selected by default in case no provider could be determined automatically.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_default_shipping_provider', + 'default' => '', + 'type' => 'select', + 'options' => wc_gzd_get_shipping_provider_select(), + 'class' => 'wc-enhanced-select', + ), + + array( + 'type' => 'sectionend', + 'id' => 'shipments_options', + ), + + array( + 'title' => _x( 'Automation', 'shipments', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'shipments_auto_options', + ), + + array( + 'title' => _x( 'Enable', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Automatically create shipments for orders.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_auto_enable', + 'default' => 'yes', + 'type' => 'gzd_toggle', + ), + + array( + 'title' => _x( 'Order statuses', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Create shipments as soon as the order reaches one of the following status(es).', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_auto_statuses', + 'default' => array( 'wc-processing', 'wc-on-hold' ), + 'class' => 'wc-enhanced-select-nostd', + 'options' => wc_get_order_statuses(), + 'type' => 'multiselect', + 'custom_attributes' => array( + 'data-show_if_woocommerce_gzd_shipments_auto_enable' => '', + 'data-placeholder' => _x( 'On new order creation', 'shipments', 'woocommerce-germanized' ), + ), + ), + + array( + 'title' => _x( 'Default status', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Choose a default status for the automatically created shipment.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_auto_default_status', + 'default' => 'gzd-processing', + 'class' => 'wc-enhanced-select', + 'options' => $statuses, + 'type' => 'select', + 'custom_attributes' => array( + 'data-show_if_woocommerce_gzd_shipments_auto_enable' => '', + ), + ), + + array( + 'title' => _x( 'Update status', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Mark order as completed after order is fully shipped.', 'shipments', 'woocommerce-germanized' ) . '
' . _x( 'This option will automatically update the order status to completed as soon as all required shipments have been marked as shipped.', 'shipments', 'woocommerce-germanized' ) . '
', + 'id' => 'woocommerce_gzd_shipments_auto_order_shipped_completed_enable', + 'default' => 'yes', + 'type' => 'gzd_toggle', + ), + + array( + 'title' => _x( 'Mark as shipped', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Mark shipments as shipped after order completion.', 'shipments', 'woocommerce-germanized' ) . '
' . _x( 'This option will automatically update contained shipments to shipped (if possible, e.g. not yet delivered) as soon as the order was marked as completed.', 'shipments', 'woocommerce-germanized' ) . '
', + 'id' => 'woocommerce_gzd_shipments_auto_order_completed_shipped_enable', + 'default' => 'no', + 'type' => 'gzd_toggle', + ), + + array( + 'type' => 'sectionend', + 'id' => 'shipments_auto_options', + ), + + array( + 'title' => _x( 'Returns', 'shipments', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'shipments_return_options', + 'desc' => sprintf( _x( 'Returns can be added manually by the shop manager or by the customer. Decide what suits you best by turning customer-added returns on or off in your %s.', 'shipments', 'woocommerce-germanized' ), '' . _x( 'shipping provider settings', 'shipments', 'woocommerce-germanized' ) . '' ), + ), + + array( + 'type' => 'shipment_return_reasons', + ), + + array( + 'title' => _x( 'Days to return', 'shipments', 'woocommerce-germanized' ), + 'desc' => '
' . sprintf( _x( 'In case one of your %s supports returns added by customers you might want to limit the number of days a customer is allowed to add returns to an order. The days are counted starting with the date the order was shipped, completed or created (by checking for existance in this order).', 'shipments', 'woocommerce-germanized' ), '' . _x( 'shipping providers', 'shipments', 'woocommerce-germanized' ) . '' ) . '
', + 'css' => 'max-width: 60px;', + 'type' => 'number', + 'id' => 'woocommerce_gzd_shipments_customer_return_open_days', + 'default' => '14', + ), + + array( + 'type' => 'sectionend', + 'id' => 'shipments_return_options', + ), + + array( + 'title' => _x( 'Customer Account', 'shipments', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'shipments_customer_options', + ), + + array( + 'title' => _x( 'List', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'List shipments on customer account order screen.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_customer_account_enable', + 'default' => 'yes', + 'type' => 'gzd_toggle', + ), + + array( + 'type' => 'sectionend', + 'id' => 'shipments_customer_options', + ), + ); + + return $settings; + } + + public static function get_address_label_by_prop( $prop, $type = 'shipper' ) { + $label = ''; + $fields = wc_gzd_get_shipment_setting_default_address_fields( $type ); + + if ( array_key_exists( $prop, $fields ) ) { + $label = $fields[ $prop ]; + } + + return $label; + } + + protected static function get_address_field_type_by_prop( $prop ) { + $type = 'text'; + + if ( 'country' === $prop ) { + $type = 'single_select_country'; + } + + return $type; + } + + protected static function get_address_desc_by_prop( $prop ) { + $desc = false; + + if ( 'customs_reference_number' === $prop ) { + $desc = _x( 'Your customs reference number, e.g. EORI number', 'shipments', 'woocommerce-germanized' ); + } elseif ( 'customs_uk_vat_id' === $prop ) { + $desc = _x( 'Your UK VAT ID, e.g. for UK exports <= 135 GBP.', 'shipments', 'woocommerce-germanized' ); + } + + return $desc; + } + + protected static function get_address_fields_to_skip() { + return array( 'state', 'street', 'street_number', 'full_name' ); + } + + protected static function get_address_settings() { + $shipper_fields = wc_gzd_get_shipment_setting_address_fields( 'shipper' ); + $return_fields = wc_gzd_get_shipment_setting_address_fields( 'return' ); + + $settings = array( + array( + 'title' => _x( 'Shipper Address', 'shipments', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'shipments_shipper_address', + ), + ); + + // Use WooCommerce address data as fallback/default data on install + $default_shipper_address_data = array( + 'company' => get_option( 'name' ), + 'address_1' => get_option( 'woocommerce_store_address' ), + 'address_2' => get_option( 'woocommerce_store_address_2' ), + 'city' => get_option( 'woocommerce_store_city' ), + 'postcode' => get_option( 'woocommerce_store_postcode' ), + 'email' => get_option( 'woocommerce_email_from_address' ), + 'country' => get_option( 'woocommerce_default_country' ), + ); + + foreach ( $shipper_fields as $field => $value ) { + if ( in_array( $field, self::get_address_fields_to_skip(), true ) ) { + continue; + } + + $default_value = ''; + + if ( array_key_exists( $field, $default_shipper_address_data ) ) { + $default_value = $default_shipper_address_data[ $field ]; + } + + $settings = array_merge( + $settings, + array( + array( + 'title' => self::get_address_label_by_prop( $field ), + 'type' => self::get_address_field_type_by_prop( $field ), + 'id' => "woocommerce_gzd_shipments_shipper_address_{$field}", + 'default' => $default_value, + 'desc_tip' => self::get_address_desc_by_prop( $field ), + ), + ) + ); + } + + $settings = array_merge( + $settings, + array( + array( + 'type' => 'sectionend', + 'id' => 'shipments_shipper_address', + ), + + array( + 'title' => _x( 'Return Address', 'shipments', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'shipments_return_address', + ), + ) + ); + + foreach ( $return_fields as $field => $value ) { + if ( in_array( $field, self::get_address_fields_to_skip(), true ) ) { + continue; + } + + $settings = array_merge( + $settings, + array( + array( + 'title' => self::get_address_label_by_prop( $field ), + 'type' => self::get_address_field_type_by_prop( $field ), + 'id' => "woocommerce_gzd_shipments_return_address_{$field}", + 'default' => '', + 'placeholder' => $value, + 'desc_tip' => self::get_address_desc_by_prop( $field ), + ), + ) + ); + } + + $settings = array_merge( + $settings, + array( + array( + 'type' => 'sectionend', + 'id' => 'shipments_return_address', + ), + ) + ); + + return $settings; + } + + protected static function get_packaging_settings() { + $settings = array( + array( + 'title' => '', + 'type' => 'title', + 'id' => 'packaging_options', + ), + + array( + 'type' => 'packaging_list', + ), + + array( + 'title' => _x( 'Default packaging', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Choose a packaging which serves as fallback or default in case no suitable packaging could be matched for a certain shipment.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'woocommerce_gzd_shipments_default_packaging', + 'default' => '', + 'type' => 'select', + 'options' => wc_gzd_get_packaging_select(), + 'class' => 'wc-enhanced-select', + ), + + array( + 'type' => 'packaging_reports', + 'title' => _x( 'Packaging Report', 'shipments', 'woocommerce-germanized' ), + 'id' => 'packaging_reports', + ), + + array( + 'type' => 'sectionend', + 'id' => 'packaging_options', + ), + ); + + return $settings; + } + + public static function get_settings( $current_section = '' ) { + $settings = array(); + + if ( '' === $current_section ) { + $settings = self::get_general_settings(); + } elseif ( 'packaging' === $current_section ) { + $settings = self::get_packaging_settings(); + } elseif ( 'address' === $current_section ) { + $settings = self::get_address_settings(); + } + + return $settings; + } + + public static function get_additional_breadcrumb_items( $breadcrumb ) { + return $breadcrumb; + } + + public static function get_sections() { + return array( + '' => _x( 'General', 'shipments', 'woocommerce-germanized' ), + 'packaging' => _x( 'Packaging', 'shipments', 'woocommerce-germanized' ), + 'address' => _x( 'Addresses', 'shipments', 'woocommerce-germanized' ), + ); + } + + public static function after_save( $current_section = '' ) { + if ( 'packaging' === $current_section ) { + if ( isset( $_POST['save'] ) && 'create_report' === wc_clean( wp_unslash( $_POST['save'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $start_date = isset( $_POST['report_year'] ) ? wc_clean( wp_unslash( $_POST['report_year'] ) ) : '01-01-' . ( (int) date( 'Y' ) - 1 ); // phpcs:ignore WordPress.Security.NonceVerification.Missing, WordPress.DateTime.RestrictedFunctions.date_date + $start_date = ReportHelper::string_to_datetime( $start_date ); + + ReportQueue::start( 'yearly', $start_date ); + } + } + } + + public static function get_sanitized_settings( $settings, $data = null ) { + if ( is_null( $data ) ) { + $data = $_POST; // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( empty( $data ) ) { + return false; + } + + $settings_to_save = array(); + + // Loop options and get values to save. + foreach ( $settings as $option ) { + + if ( ! isset( $option['id'] ) || empty( $option['id'] ) || ! isset( $option['type'] ) || in_array( $option['type'], array( 'title', 'sectionend' ), true ) || ( isset( $option['is_option'] ) && false === $option['is_option'] ) ) { + continue; + } + + $option_key = $option['id']; + $raw_value = isset( $data[ $option_key ] ) ? wp_unslash( $data[ $option_key ] ) : null; + + // Format the value based on option type. + switch ( $option['type'] ) { + case 'checkbox': + $value = '1' === $raw_value || 'yes' === $raw_value ? 'yes' : 'no'; + break; + case 'textarea': + $value = wp_kses_post( trim( $raw_value ) ); + break; + case 'password': + $value = is_null( $raw_value ) ? '' : addslashes( $raw_value ); + $value = trim( $value ); + + if ( class_exists( 'WC_GZD_Secret_Box_Helper' ) ) { + $encrypted = \WC_GZD_Secret_Box_Helper::encrypt( $value ); + + if ( ! is_wp_error( $encrypted ) ) { + $value = $encrypted; + } + } + break; + case 'multiselect': + case 'multi_select_countries': + $value = array_filter( array_map( 'wc_clean', (array) $raw_value ) ); + break; + case 'image_width': + $value = array(); + if ( isset( $raw_value['width'] ) ) { + $value['width'] = wc_clean( $raw_value['width'] ); + $value['height'] = wc_clean( $raw_value['height'] ); + $value['crop'] = isset( $raw_value['crop'] ) ? 1 : 0; + } else { + $value['width'] = $option['default']['width']; + $value['height'] = $option['default']['height']; + $value['crop'] = $option['default']['crop']; + } + break; + case 'select': + $allowed_values = empty( $option['options'] ) ? array() : array_map( 'strval', array_keys( $option['options'] ) ); + if ( empty( $option['default'] ) && empty( $allowed_values ) ) { + $value = null; + break; + } + $default = ( empty( $option['default'] ) ? $allowed_values[0] : $option['default'] ); + $value = in_array( $raw_value, $allowed_values, true ) ? $raw_value : $default; + break; + case 'relative_date_selector': + $value = wc_parse_relative_date_option( $raw_value ); + break; + default: + $value = wc_clean( $raw_value ); + break; + } + + /** + * Sanitize the value of an option. + * + * @since 2.4.0 + */ + $value = apply_filters( 'woocommerce_admin_settings_sanitize_option', $value, $option, $raw_value ); + + $settings_to_save[ $option_key ] = $value; + } + + return $settings_to_save; + } + + public static function render_label_fields( $settings, $shipment, $echo = false ) { + $missing_div_closes = 0; + ob_start(); + foreach ( $settings as $setting ) { + $setting = wp_parse_args( + $setting, + array( + 'id' => '', + 'type' => 'text', + 'custom_attributes' => array(), + ) + ); + + if ( has_action( "woocommerce_gzd_shipment_label_admin_field_{$setting['id']}" ) ) { + do_action( "woocommerce_gzd_shipment_label_admin_field_{$setting['id']}", $setting, $shipment ); + } elseif ( 'select' === $setting['type'] ) { + woocommerce_wp_select( $setting ); + } elseif ( 'multiselect' === $setting['type'] ) { + $setting['class'] = 'select short wc-enhanced-select'; + $setting['custom_attributes'] = array_merge( $setting['custom_attributes'], array( 'multiple' => 'multiple' ) ); + + if ( ! strstr( $setting['id'], '[]' ) ) { + $setting['name'] = $setting['id'] . '[]'; + } + + woocommerce_wp_select( $setting ); + } elseif ( 'checkbox' === $setting['type'] ) { + woocommerce_wp_checkbox( $setting ); + } elseif ( 'textarea' === $setting['type'] ) { + woocommerce_wp_textarea_input( $setting ); + } elseif ( 'text' === $setting['type'] ) { + woocommerce_wp_text_input( $setting ); + } elseif ( 'date' === $setting['type'] ) { + $setting['class'] = 'datepicker'; + $setting['type'] = 'date'; + + woocommerce_wp_text_input( $setting ); + } elseif ( 'number' === $setting['type'] ) { + woocommerce_wp_text_input( $setting ); + } elseif ( 'services_start' === $setting['type'] ) { + $hide_default = isset( $setting['hide_default'] ) ? wc_string_to_bool( $setting['hide_default'] ) : false; + $missing_div_closes++; + ?> +

+ + + + + + +

+
+ +
+ +
+ +
+ 0 ) { + while ( $missing_div_closes > 0 ) { + $missing_div_closes--; + echo '
'; + } + } + + $html = ob_get_clean(); + + if ( ! $echo ) { + return $html; + } else { + echo $html; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Admin/Table.php b/packages/woocommerce-germanized-shipments/src/Admin/Table.php new file mode 100644 index 000000000..6b5ca0eb4 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Admin/Table.php @@ -0,0 +1,1200 @@ + 'simple', + ) + ); + + $this->shipment_type = $args['type']; + + parent::__construct( + array( + 'plural' => 'shipments', + 'screen' => isset( $args['screen'] ) ? $args['screen'] : null, + ) + ); + } + + public function set_default_hidden_columns( $columns, $screen ) { + if ( $this->screen->id === $screen->id ) { + $columns = array_merge( $columns, $this->get_default_hidden_columns() ); + } + + return $columns; + } + + protected function get_default_hidden_columns() { + return array( + 'weight', + 'dimensions', + 'packaging', + ); + } + + public function enable_query_removing( $args ) { + $args = array_merge( + $args, + array( + 'changed', + 'bulk_action', + ) + ); + + return $args; + } + + /** + * Handle bulk actions. + * + * @param string $redirect_to URL to redirect to. + * @param string $action Action name. + * @param array $ids List of ids. + * @return string + */ + public function handle_bulk_actions( $action, $ids, $redirect_to ) { + $ids = array_reverse( array_map( 'absint', $ids ) ); + $changed = 0; + + if ( false !== strpos( $action, 'mark_' ) ) { + + $shipment_statuses = wc_gzd_get_shipment_statuses(); + $new_status = substr( $action, 5 ); // Get the status name from action. + + // Sanity check: bail out if this is actually not a status, or is not a registered status. + if ( isset( $shipment_statuses[ 'gzd-' . $new_status ] ) ) { + + foreach ( $ids as $id ) { + + if ( $shipment = wc_gzd_get_shipment( $id ) ) { + $shipment->update_status( $new_status, true ); + + /** + * Action that fires after a shipment bulk status update has been processed. + * + * @param integer $shipment_id The shipment id. + * @param string $new_status The new shipment status. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_edit_status', $id, $new_status ); + $changed++; + } + } + } + } elseif ( 'delete' === $action ) { + foreach ( $ids as $id ) { + if ( $shipment = wc_gzd_get_shipment( $id ) ) { + $shipment->delete( true ); + $changed++; + } + } + } elseif ( 'confirm_requests' === $action ) { + foreach ( $ids as $id ) { + if ( $shipment = wc_gzd_get_shipment( $id ) ) { + if ( 'return' === $shipment->get_type() ) { + if ( $shipment->is_customer_requested() && $shipment->has_status( 'requested' ) ) { + if ( $shipment->confirm_customer_request() ) { + $changed++; + } + } + } + } + } + } + + /** + * Filter to decide whether a Shipment has changed during bulk action or not. + * + * @param boolean $changed Whether the Shipment has changed or not. + * @param string $action The bulk action + * @param string $redirect_to The redirect URL. + * @param Table $table The table instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $changed = apply_filters( 'woocommerce_gzd_shipments_bulk_action', $changed, $action, $ids, $redirect_to, $this ); + + if ( $changed ) { + $redirect_to = add_query_arg( + array( + 'changed' => $changed, + 'ids' => join( ',', $ids ), + 'bulk_action' => $action, + ), + $redirect_to + ); + } + + return esc_url_raw( $redirect_to ); + } + + public function output_notice() { + + if ( ! empty( $this->notice ) ) { + $type = isset( $this->notice['type'] ) ? $this->notice['type'] : 'success'; + + echo '
' . wp_kses_post( wpautop( $this->notice['message'] ) ) . '
'; + } + + $this->notice = array(); + } + + /** + * Show confirmation message that order status changed for number of orders. + */ + public function set_bulk_notice() { + + $number = isset( $_REQUEST['changed'] ) ? absint( $_REQUEST['changed'] ) : 0; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $bulk_action = isset( $_REQUEST['bulk_action'] ) ? wc_clean( wp_unslash( $_REQUEST['bulk_action'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + if ( 'delete' === $bulk_action ) { + + $this->set_notice( sprintf( _nx( '%d shipment deleted.', '%d shipments deleted.', $number, 'shipments', 'woocommerce-germanized' ), number_format_i18n( $number ) ) ); + + } elseif ( strpos( $bulk_action, 'mark_' ) !== false ) { + + $shipment_statuses = wc_gzd_get_shipment_statuses(); + + // Check if any status changes happened. + foreach ( $shipment_statuses as $slug => $name ) { + + if ( 'mark_' . str_replace( 'gzd-', '', $slug ) === $bulk_action ) { // WPCS: input var ok, CSRF ok. + $this->set_notice( sprintf( _nx( '%d shipment status changed.', '%d shipment statuses changed.', $number, 'shipments', 'woocommerce-germanized' ), number_format_i18n( $number ) ) ); + break; + } + } + } + + /** + * Action that fires after bulk updating shipments. Action might be usefull to add + * custom notices after custom bulk actions have been applied. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_bulk_notice + * + * @param string $bulk_action The bulk action. + * @param Table $shipment_table The table object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$this->get_hook_prefix()}bulk_notice", $bulk_action, $this ); + } + + public function set_notice( $message, $type = 'success' ) { + $this->notice = array( + 'message' => $message, + 'type' => $type, + ); + } + + protected function get_stati() { + return $this->stati; + } + + /** + * @return bool + */ + public function ajax_user_can() { + return current_user_can( 'edit_shop_orders' ); + } + + public function get_page_option() { + return 'woocommerce_page_wc_gzd_shipments_per_page'; + } + + /** + * @global array $avail_post_stati + * @global WP_Query $wp_query + * @global int $per_page + * @global string $mode + */ + public function prepare_items() { + global $per_page; + + $per_page = $this->get_items_per_page( $this->get_page_option(), 10 ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited + + /** + * Filter to adjust Shipment's table items per page. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_edit_per_page + * + * @param integer $per_page Number of Shipments per page. + * @param string $type The type in this case shipment. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $per_page = apply_filters( "{$this->get_hook_prefix()}edit_per_page", $per_page, 'shipment' ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited + $this->stati = wc_gzd_get_shipment_statuses(); + $this->counts = wc_gzd_get_shipment_counts( $this->shipment_type ); + $paged = $this->get_pagenum(); + + $args = array( + 'limit' => $per_page, + 'paginate' => true, + 'offset' => ( $paged - 1 ) * $per_page, + 'count_total' => true, + 'type' => $this->shipment_type, + ); + + if ( isset( $_REQUEST['shipment_status'] ) && in_array( $_REQUEST['shipment_status'], array_keys( $this->stati ), true ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $args['status'] = wc_clean( wp_unslash( $_REQUEST['shipment_status'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + } + + if ( isset( $_REQUEST['orderby'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + if ( 'weight' === $_REQUEST['orderby'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $args['orderby'] = 'weight'; + } else { + $args['orderby'] = wc_clean( wp_unslash( $_REQUEST['orderby'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + } + } + + if ( isset( $_REQUEST['order'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $args['order'] = 'asc' === $_REQUEST['order'] ? 'ASC' : 'DESC'; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + } + + if ( isset( $_REQUEST['parent_id'] ) && ! empty( $_REQUEST['parent_id'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $args['parent_id'] = absint( $_REQUEST['parent_id'] ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + } + + if ( isset( $_REQUEST['order_id'] ) && ! empty( $_REQUEST['order_id'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $args['order_id'] = absint( $_REQUEST['order_id'] ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + } + + if ( isset( $_REQUEST['shipping_provider'] ) && ! empty( $_REQUEST['shipping_provider'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $args['shipping_provider'] = wc_clean( wp_unslash( $_REQUEST['shipping_provider'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + } + + if ( isset( $_REQUEST['m'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $m = wc_clean( wp_unslash( $_REQUEST['m'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $year = substr( $m, 0, 4 ); + + if ( ! empty( $year ) ) { + $month = ''; + $day = ''; + + if ( strlen( $m ) > 5 ) { + $month = substr( $m, 4, 2 ); + } + + if ( strlen( $m ) > 7 ) { + $day = substr( $m, 6, 2 ); + } + + $datetime = new WC_DateTime(); + $datetime->setDate( $year, 1, 1 ); + + if ( ! empty( $month ) ) { + $datetime->setDate( $year, $month, 1 ); + } + + if ( ! empty( $day ) ) { + $datetime->setDate( $year, $month, $day ); + } + + $next_month = clone $datetime; + $next_month->modify( '+ 1 month' ); + // Make sure to not include next month first day + $next_month->modify( '-1 day' ); + + $args['date_created'] = $datetime->format( 'Y-m-d' ) . '...' . $next_month->format( 'Y-m-d' ); + } + } + + if ( isset( $_REQUEST['s'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $search = wc_clean( wp_unslash( $_REQUEST['s'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + if ( ! is_numeric( $search ) ) { + $search = '*' . $search . '*'; + } + + $args['search'] = $search; + } + + // Query the user IDs for this page + $this->query = new ShipmentQuery( apply_filters( "{$this->get_hook_prefix()}query_args", $args, $this ) ); + $this->items = $this->query->get_shipments(); + + $this->set_pagination_args( + array( + 'total_items' => $this->query->get_total(), + 'per_page' => $per_page, + ) + ); + } + + /** + */ + public function no_items() { + echo esc_html_x( 'No shipments found', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Determine if the current view is the "All" view. + * + * @since 4.2.0 + * + * @return bool Whether the current view is the "All" view. + */ + protected function is_base_request() { + $vars = $_GET; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + unset( $vars['paged'] ); + + if ( empty( $vars ) ) { + return true; + } + + return 1 === count( $vars ); + } + + /** + * @global array $locked_post_status This seems to be deprecated. + * @global array $avail_post_stati + * @return array + */ + protected function get_views() { + + $status_links = array(); + $num_shipments = $this->counts; + $total_shipments = array_sum( (array) $num_shipments ); + $class = ''; + $all_args = array(); + + if ( empty( $class ) && ( $this->is_base_request() || isset( $_REQUEST['all_shipments'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $class = 'current'; + } + + $all_inner_html = sprintf( + _nx( + 'All (%s)', + 'All (%s)', + $total_shipments, + 'shipments', + 'woocommerce-germanized-shipments' + ), + number_format_i18n( $total_shipments ) + ); + + $status_links['all'] = $this->get_edit_link( $all_args, $all_inner_html, $class ); + + foreach ( wc_gzd_get_shipment_statuses() as $status => $title ) { + $class = ''; + + if ( ! in_array( $status, array_keys( $this->stati ), true ) || empty( $num_shipments[ $status ] ) ) { + continue; + } + + if ( isset( $_REQUEST['shipment_status'] ) && $status === $_REQUEST['shipment_status'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $class = 'current'; + } + + $status_args = array( + 'shipment_status' => $status, + ); + + $status_label = sprintf( + translate_nooped_plural( _nx_noop( ( $title . ' (%s)' ), ( $title . ' (%s)' ), 'shipments', 'woocommerce-germanized' ), $num_shipments[ $status ] ), // phpcs:ignore WordPress.WP.I18n.NonSingularStringLiteralSingle,WordPress.WP.I18n.NonSingularStringLiteralPlural + number_format_i18n( $num_shipments[ $status ] ) + ); + + $status_links[ $status ] = $this->get_edit_link( $status_args, $status_label, $class ); + } + + return $status_links; + } + + /** + * Helper to create links to edit.php with params. + * + * @since 4.4.0 + * + * @param string[] $args Associative array of URL parameters for the link. + * @param string $label Link text. + * @param string $class Optional. Class attribute. Default empty string. + * @return string The formatted link string. + */ + protected function get_edit_link( $args, $label, $class = '' ) { + $url = add_query_arg( $args, $this->get_main_page() ); + + $class_html = $aria_current = ''; + if ( ! empty( $class ) ) { + $class_html = sprintf( + ' class="%s"', + esc_attr( $class ) + ); + + if ( 'current' === $class ) { + $aria_current = ' aria-current="page"'; + } + } + + return sprintf( + '%s', + esc_url( $url ), + $class_html, + $aria_current, + $label + ); + } + + /** + * @return string + */ + public function current_action() { + if ( isset( $_REQUEST['delete_all'] ) || isset( $_REQUEST['delete_all2'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + return 'delete_all'; + } + + return parent::current_action(); + } + + /** + * Display a monthly dropdown for filtering items + * + * @since 3.0.6 + * + * @global wpdb $wpdb + * @global WP_Locale $wp_locale + * + * @param string $post_type + */ + protected function months_dropdown( $type ) { + global $wpdb, $wp_locale; + + $extra_checks = "AND shipment_status != 'auto-draft'"; + + if ( isset( $_GET['shipment_status'] ) && 'all' !== $_GET['shipment_status'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $extra_checks = $wpdb->prepare( ' AND shipment_status = %s', wc_clean( wp_unslash( $_GET['shipment_status'] ) ) ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + } + + $months = $wpdb->get_results( "SELECT DISTINCT YEAR( shipment_date_created ) AS year, MONTH( shipment_date_created ) AS month FROM $wpdb->gzd_shipments WHERE 1=1 $extra_checks ORDER BY shipment_date_created DESC" ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared + $month_count = count( $months ); + + if ( ! $month_count || ( 1 === $month_count && 0 === $months[0]->month ) ) { + return; + } + + $m = isset( $_GET['m'] ) ? absint( wp_unslash( $_GET['m'] ) ) : 0; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + ?> + + + +
+

+
+ +
+ get_notices( 'error' ); + $info = $handler->get_notices( 'info' ); + $success = $handler->get_notices( 'success' ); + ?> + + +
+

+
+ + + admin_after_error(); ?> + + + +
+

+
+ + + +
+

get_success_message() ); ?>

+
+ + + admin_handled(); + /** + * Action that fires after a certain bulk action result has been rendered. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * `$bulk_action` refers to the bulk action handled. + * + * Example hook name: woocommerce_gzd_return_shipments_table_mark_processing_handled + * + * @param BulkActionHandler $bulk_action_handler The bulk action handler. + * @param string $bulk_action The bulk action. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$this->get_hook_prefix()}bulk_action_{$bulk_action}_handled", $handler, $bulk_action ); + ?> + + reset(); ?> + + +
+ months_dropdown( 'shipment' ); + $this->order_filter(); + $this->shipping_provider_filter(); + + /** + * Action that fires after outputting Shipments table view filters. + * Might be used to add custom filters to the Shipments table view. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_filters + * + * @param string $which top or bottom. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$this->get_hook_prefix()}filters", $which ); + + $output = ob_get_clean(); + + if ( ! empty( $output ) ) { + echo $output; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + + submit_button( _x( 'Filter', 'shipments', 'woocommerce-germanized' ), '', 'filter_action', false, array( 'id' => 'shipment-query-submit' ) ); + } + } + ?> +
+ get_title(); + } + } + ?> + + + + '; + $columns['title'] = _x( 'Title', 'shipments', 'woocommerce-germanized' ); + $columns['date'] = _x( 'Date', 'shipments', 'woocommerce-germanized' ); + $columns['status'] = _x( 'Status', 'shipments', 'woocommerce-germanized' ); + $columns['items'] = _x( 'Items', 'shipments', 'woocommerce-germanized' ); + $columns['address'] = _x( 'Address', 'shipments', 'woocommerce-germanized' ); + $columns['packaging'] = _x( 'Packaging', 'shipments', 'woocommerce-germanized' ); + $columns['weight'] = _x( 'Weight', 'shipments', 'woocommerce-germanized' ); + $columns['dimensions'] = _x( 'Dimensions', 'shipments', 'woocommerce-germanized' ); + $columns['order'] = _x( 'Order', 'shipments', 'woocommerce-germanized' ); + $columns['actions'] = _x( 'Actions', 'shipments', 'woocommerce-germanized' ); + + return $columns; + } + + /** + * @return array + */ + public function get_columns() { + $columns = $this->get_custom_columns(); + + /** + * Filters the columns displayed in the Shipments list table. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_edit_per_page + * + * @param string[] $columns An associative array of column headings. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $columns = apply_filters( "{$this->get_hook_prefix()}columns", $columns ); + + return $columns; + } + + /** + * @return array + */ + protected function get_sortable_columns() { + return apply_filters( + "{$this->get_hook_prefix()}sortable_columns", + array( + 'date' => array( 'date_created', false ), + 'weight' => 'weight', + 'order' => 'order_id', + ), + $this + ); + } + + /** + * Gets the name of the default primary column. + * + * @since 4.3.0 + * + * @return string Name of the default primary column, in this case, 'title'. + */ + protected function get_default_primary_column_name() { + return 'title'; + } + + /** + * Handles the default column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + * @param string $column_name The current column name. + */ + public function column_default( $shipment, $column_name ) { + + /** + * Fires in each custom column in the Shipments list table. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_filters + * + * @param string $column_name The name of the column to display. + * @param integer $shipment_id The current shipment id. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$this->get_hook_prefix()}custom_column", $column_name, $shipment->get_id() ); + } + + public function get_main_page() { + return 'admin.php?page=wc-gzd-shipments'; + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + */ + public function column_title( $shipment ) { + $title = sprintf( _x( '%1$s #%2$s', 'shipment title', 'woocommerce-germanized' ), wc_gzd_get_shipment_label_title( $shipment->get_type() ), $shipment->get_shipment_number() ); + + if ( $order = $shipment->get_order() ) { + echo '' . wp_kses_post( $title ) . ' '; + } else { + echo wp_kses_post( $title ) . ' '; + } + + echo '

'; + + if ( $packaging = $shipment->get_packaging() ) { + echo '' . wp_kses_post( $packaging->get_description() ) . ' '; + } + + $provider = $shipment->get_shipping_provider(); + + if ( ! empty( $provider ) ) { + echo '' . sprintf( esc_html_x( 'via %s', 'shipments', 'woocommerce-germanized' ), wp_kses_post( wc_gzd_get_shipping_provider_title( $provider ) ) ) . ' '; + } + + if ( $tracking_id = $shipment->get_tracking_id() ) { + if ( $shipment->has_tracking() && ( $tracking_url = $shipment->get_tracking_url() ) ) { + echo '' . esc_html( $tracking_id ) . ''; + } else { + echo '' . esc_html( $tracking_id ) . ''; + } + } + + echo '

'; + } + + protected function get_custom_actions( $shipment, $actions ) { + return $actions; + } + + /** + * Handles shipment actions. + * + * @since 0.0.1 + * + * @param Shipment $shipment The current shipment object. + */ + protected function column_actions( $shipment ) { + echo '

'; + + /** + * Action that fires before table actions are outputted for a Shipment. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_actions_start + * + * @param Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$this->get_hook_prefix()}actions_start", $shipment ); + + $actions = array(); + + if ( $shipment->has_status( array( 'draft' ) ) ) { + $actions['processing'] = array( + 'url' => wp_nonce_url( admin_url( 'admin-ajax.php?action=woocommerce_gzd_update_shipment_status&status=processing&shipment_id=' . $shipment->get_id() ), 'update-shipment-status' ), + 'name' => _x( 'Processing', 'shipments', 'woocommerce-germanized' ), + 'action' => 'processing', + ); + } + + if ( $shipment->has_status( array( 'draft', 'processing' ) ) ) { + $actions['shipped'] = array( + 'url' => wp_nonce_url( admin_url( 'admin-ajax.php?action=woocommerce_gzd_update_shipment_status&status=shipped&shipment_id=' . $shipment->get_id() ), 'update-shipment-status' ), + 'name' => _x( 'Shipped', 'shipments', 'woocommerce-germanized' ), + 'action' => 'shipped', + ); + } + + if ( $shipment->supports_label() ) { + + if ( $label = $shipment->get_label() ) { + + $actions['download_label'] = array( + 'url' => $label->get_download_url(), + 'name' => _x( 'Download label', 'shipments', 'woocommerce-germanized' ), + 'action' => 'download-label download', + 'target' => '_blank', + ); + + } elseif ( $shipment->needs_label() ) { + + $actions['generate_label'] = array( + 'url' => '#', + 'name' => _x( 'Generate label', 'shipments', 'woocommerce-germanized' ), + 'action' => 'generate-label generate', + ); + + include Package::get_path() . '/includes/admin/views/label/html-shipment-label-backbone.php'; + } + } + + $actions = $this->get_custom_actions( $shipment, $actions ); + + /** + * Filters the actions available for Shipments table list column. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_actions + * + * @param array $actions The registered Shipment actions. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $actions = apply_filters( "{$this->get_hook_prefix()}actions", $actions, $shipment ); + + echo wc_gzd_render_shipment_action_buttons( $actions ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + + /** + * Action that fires after table actions are outputted for a Shipment. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_actions_end + * + * @param Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$this->get_hook_prefix()}actions_end", $shipment ); + + echo '

'; + } + + public function column_cb( $shipment ) { + if ( current_user_can( 'edit_shop_orders' ) ) : + ?> + + + + + + get_items() as $item ) : ?> + + + + + + +
+ get_product() ) : ?> + get_name() ); ?> + + get_name() ); ?> + + + get_sku() ? '
' . esc_html_x( 'SKU:', 'shipments', 'woocommerce-germanized' ) . ' ' . esc_html( $item->get_sku() ) . '' : '' ); ?> + + get_hook_prefix()}item_after_name", $item->get_id(), $item, $shipment ); + ?> +
+ get_quantity() ); ?>x +
+ get_formatted_address(); + + if ( $address ) { + echo '' . esc_html( preg_replace( '##i', ', ', $address ) ) . ''; + } else { + echo '–'; + } + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + */ + public function column_status( $shipment ) { + echo '' . esc_html( wc_gzd_get_shipment_status_name( $shipment->get_status() ) ) . ''; + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + */ + public function column_weight( $shipment ) { + echo wc_gzd_format_shipment_weight( $shipment->get_total_weight(), $shipment->get_weight_unit() ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + */ + public function column_packaging( $shipment ) { + if ( $packaging = $shipment->get_packaging() ) { + echo wp_kses_post( $packaging->get_description() ); + } else { + echo '–'; + } + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + */ + public function column_dimensions( $shipment ) { + echo wc_gzd_format_shipment_dimensions( $shipment->get_dimensions(), $shipment->get_dimension_unit() ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + */ + public function column_date( $shipment ) { + $shipment_timestamp = $shipment->get_date_created() ? $shipment->get_date_created()->getTimestamp() : ''; + + if ( ! $shipment_timestamp ) { + echo '–'; + return; + } + + // Check if the order was created within the last 24 hours, and not in the future. + if ( $shipment_timestamp > strtotime( '-1 day', time() ) && $shipment_timestamp <= time() ) { + $show_date = sprintf( + /* translators: %s: human-readable time difference */ + _x( '%s ago', '%s = human-readable time difference', 'woocommerce-germanized' ), + human_time_diff( $shipment->get_date_created()->getTimestamp(), time() ) + ); + } else { + /** + * Filter to adjust the Shipment date format in table view. + * + * @param string $format The date format. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $show_date = $shipment->get_date_created()->date_i18n( apply_filters( 'woocommerce_gzd_admin_shipment_date_format', _x( 'M j, Y', 'shipments', 'woocommerce-germanized' ) ) ); + } + + printf( + '', + esc_attr( $shipment->get_date_created()->date( 'c' ) ), + esc_html( $shipment->get_date_created()->date_i18n( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ) ) ), + esc_html( $show_date ) + ); + } + + /** + * Handles the post author column output. + * + * @since 4.3.0 + * + * @param Shipment $shipment The current shipment object. + */ + public function column_order( $shipment ) { + if ( ( $order = $shipment->get_order() ) && is_callable( array( $order, 'get_edit_order_url' ) ) ) { + echo '' . esc_html( $order->get_order_number() ) . ''; + } else { + echo esc_html( $shipment->get_order_id() ); + } + } + + /** + * + * @param int|WC_GZD_Shipment $shipment + */ + public function single_row( $shipment ) { + $GLOBALS['shipment'] = $shipment; + $classes = 'shipment shipment-status-' . $shipment->get_status(); + ?> + + single_row_columns( $shipment ); ?> + + shipment_type ? '' : '_' . $this->shipment_type ); + + return "woocommerce_gzd{$suffix}_shipments_table_"; + } + + /** + * @return array + */ + protected function get_bulk_actions() { + $actions = array(); + + if ( current_user_can( 'delete_shop_orders' ) ) { + $actions['delete'] = _x( 'Delete Permanently', 'shipments', 'woocommerce-germanized' ); + } + + $actions['mark_processing'] = _x( 'Change status to processing', 'shipments', 'woocommerce-germanized' ); + $actions['mark_shipped'] = _x( 'Change status to shipped', 'shipments', 'woocommerce-germanized' ); + $actions['mark_delivered'] = _x( 'Change status to delivered', 'shipments', 'woocommerce-germanized' ); + $actions['labels'] = _x( 'Generate and download labels', 'shipments', 'woocommerce-germanized' ); + + $actions = $this->get_custom_bulk_actions( $actions ); + + /** + * Filter to register addtional bulk actions for shipments. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type e.g. return. In case of simple shipments the type is omitted. + * + * Example hook name: woocommerce_gzd_return_shipments_table_bulk_actions + * + * @param array $actions Array containing key => value pairs. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}bulk_actions", $actions ); + } + +} diff --git a/packages/woocommerce-germanized-shipments/src/Ajax.php b/packages/woocommerce-germanized-shipments/src/Ajax.php new file mode 100644 index 000000000..8c7415359 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Ajax.php @@ -0,0 +1,1558 @@ +hide_errors(); + } + + public static function send_return_shipment_notification_email() { + $success = false; + + if ( current_user_can( 'edit_shop_orders' ) && isset( $_REQUEST['shipment_id'] ) ) { + + if ( isset( $_GET['shipment_id'] ) ) { + $referrer = check_admin_referer( 'send-return-shipment-notification' ); + } else { + $referrer = check_ajax_referer( 'send-return-shipment-notification', 'security' ); + } + + if ( $referrer ) { + $shipment_id = absint( wp_unslash( $_REQUEST['shipment_id'] ) ); + + if ( $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + + if ( 'return' === $shipment->get_type() ) { + WC()->mailer()->emails['WC_GZD_Email_Customer_Return_Shipment']->trigger( $shipment_id ); + $success = true; + } + } + } + + if ( isset( $_GET['shipment_id'] ) ) { + wp_safe_redirect( wp_get_referer() ? wp_get_referer() : admin_url( 'admin.php?page=wc-gzd-return-shipments' ) ); + exit; + } else { + if ( $success ) { + wp_send_json( + array( + 'success' => true, + 'messages' => array( + _x( 'Notification successfully sent to customer.', 'shipments', 'woocommerce-germanized' ), + ), + ) + ); + } else { + wp_send_json( + array( + 'success' => false, + 'messages' => array( + _x( 'There was an error while sending the notification.', 'shipments', 'woocommerce-germanized' ), + ), + ) + ); + } + } + } + } + + public static function confirm_return_request() { + $success = false; + + if ( current_user_can( 'edit_shop_orders' ) && isset( $_REQUEST['shipment_id'] ) ) { + + if ( isset( $_GET['shipment_id'] ) ) { + $referrer = check_admin_referer( 'confirm-return-request' ); + } else { + $referrer = check_ajax_referer( 'confirm-return-request', 'security' ); + } + + if ( $referrer ) { + $shipment_id = absint( wp_unslash( $_REQUEST['shipment_id'] ) ); + + if ( $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + if ( 'return' === $shipment->get_type() ) { + + if ( $shipment->confirm_customer_request() ) { + $success = true; + } + } + } + } + + if ( isset( $_GET['shipment_id'] ) ) { + wp_safe_redirect( wp_get_referer() ? wp_get_referer() : admin_url( 'admin.php?page=wc-gzd-return-shipments' ) ); + exit; + } else { + if ( $success ) { + wp_send_json( + array( + 'success' => true, + 'messages' => array( + _x( 'Return request confirmed successfully.', 'shipments', 'woocommerce-germanized' ), + ), + 'shipment_id' => $shipment->get_id(), + 'needs_refresh' => true, + 'fragments' => array( + 'div#shipment-' . $shipment_id => self::get_shipment_html( $shipment ), + ), + ) + ); + } else { + wp_send_json( + array( + 'success' => false, + 'messages' => array( + _x( 'There was an error while confirming the request.', 'shipments', 'woocommerce-germanized' ), + ), + ) + ); + } + } + } + } + + public static function create_shipment_label_form() { + check_ajax_referer( 'create-shipment-label-form', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $shipment_id = absint( $_POST['shipment_id'] ); + $response = array(); + $response_error = array( + 'success' => false, + 'messages' => array( + _x( 'There was an error creating the label.', 'shipments', 'woocommerce-germanized' ), + ), + ); + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( $shipment->supports_label() && $shipment->needs_label() ) { + $html = $shipment->get_label_settings_html(); + } + + $response = array( + 'fragments' => array( + '.wc-gzd-shipment-create-label' => '
' . $html . '
', + ), + 'shipment_id' => $shipment_id, + 'success' => true, + ); + + wp_send_json( $response ); + } + + public static function remove_shipment_label() { + check_ajax_referer( 'remove-shipment-label', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $response = array(); + $response_error = array( + 'success' => false, + 'messages' => array( + _x( 'There was an error deleting the label.', 'shipments', 'woocommerce-germanized' ), + ), + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $label = $shipment->get_label() ) { + wp_send_json( $response_error ); + } + + if ( $shipment->delete_label( true ) ) { + $response = array( + 'success' => true, + 'shipment_id' => $shipment->get_id(), + 'needs_refresh' => true, + 'fragments' => array( + 'div#shipment-' . $shipment_id => self::get_shipment_html( $shipment ), + ), + ); + } else { + wp_send_json( $response_error ); + } + + wp_send_json( $response ); + } + + public static function create_shipment_label() { + check_ajax_referer( 'create-shipment-label', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $response = array(); + $response_error = array( + 'success' => false, + 'messages' => array( + _x( 'There was an error processing the label.', 'shipments', 'woocommerce-germanized' ), + ), + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + $result = false; + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( $shipment->supports_label() && $shipment->needs_label() ) { + $data = array(); + + foreach ( $_POST as $key => $value ) { + if ( in_array( $key, array( 'action', 'security' ), true ) ) { + continue; + } + + $data[ $key ] = wc_clean( wp_unslash( $value ) ); + } + + $result = $shipment->create_label( $data ); + } + + if ( is_wp_error( $result ) ) { + $result = wc_gzd_get_shipment_error( $result ); + } + + if ( is_wp_error( $result ) && ! $result->is_soft_error() ) { + $response = array( + 'success' => false, + 'messages' => $result->get_error_messages_by_type(), + ); + } elseif ( $label = $shipment->get_label() ) { + $order_shipment = wc_gzd_get_shipment_order( $shipment->get_order() ); + $order_status_html = $order_shipment ? self::get_global_order_status_html( $order_shipment->get_order() ) : array(); + + $response = array( + 'success' => true, + 'label_id' => $label->get_id(), + 'shipment_id' => $shipment_id, + 'messages' => is_wp_error( $result ) ? $result->get_error_messages_by_type() : array(), + 'needs_refresh' => true, + 'fragments' => array( + 'div#shipment-' . $shipment_id => self::get_shipment_html( $shipment ), + '.order-shipping-status' => $order_shipment ? self::get_order_status_html( $order_shipment ) : '', + '.order-return-status' => $order_shipment ? self::get_order_return_status_html( $order_shipment ) : '', + '.order_data_column p.wc-order-status' => ! empty( $order_status_html ) ? $order_status_html['status'] : '', + 'input[name=post_status]' => ! empty( $order_status_html ) ? $order_status_html['input'] : '', + 'tr#shipment-' . $shipment_id . ' td.actions .wc-gzd-shipment-action-button-generate-label' => self::label_download_button_html( $label ), + ), + ); + + if ( empty( $response['fragments']['.order_data_column p.wc-order-status'] ) ) { + unset( $response['fragments']['.order_data_column p.wc-order-status'] ); + } + } else { + $response = $response_error; + } + + wp_send_json( $response ); + } + + protected static function get_shipment_html( $p_shipment, $p_is_active = true ) { + $is_active = $p_is_active; + $shipment = $p_shipment; + + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-shipment.php'; + $html = ob_get_clean(); + + return $html; + } + + protected static function get_label_html( $p_shipment, $p_label = false ) { + $shipment = $p_shipment; + + if ( $p_label ) { + $label = $p_label; + } + + ob_start(); + include Package::get_path() . '/includes/admin/views/label/html-shipment-label.php'; + $html = ob_get_clean(); + + return $html; + } + + /** + * @param ShipmentLabel $label + * + * @return string + */ + protected static function label_download_button_html( $label ) { + return '' . _x( 'Download label', 'shipments', 'woocommerce-germanized' ) . ''; + } + + public static function edit_shipping_provider_status() { + check_ajax_referer( 'edit-shipping-providers', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['provider'] ) || ! isset( $_POST['enable'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error while trying to save the shipping provider status.', 'shipments', 'woocommerce-germanized' ), + ); + + $provider = sanitize_key( wc_clean( wp_unslash( $_POST['provider'] ) ) ); + $enable = wc_clean( wp_unslash( $_POST['enable'] ) ); + $helper = Helper::instance(); + $response = array( + 'success' => true, + 'provider' => $provider, + 'message' => '', + ); + + $helper->load_shipping_providers(); + + if ( $shipping_provider = $helper->get_shipping_provider( $provider ) ) { + if ( 'yes' === $enable ) { + $response['activated'] = 'yes'; + $shipping_provider->activate(); + } else { + $response['activated'] = 'no'; + $shipping_provider->deactivate(); + } + + wp_send_json( $response ); + } else { + wp_send_json( $response_error ); + } + } + + public static function remove_shipping_provider() { + check_ajax_referer( 'remove-shipping-provider', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['provider'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error while trying to delete the shipping provider.', 'shipments', 'woocommerce-germanized' ), + ); + + $provider = sanitize_key( wc_clean( wp_unslash( $_POST['provider'] ) ) ); + $helper = Helper::instance(); + $response = array( + 'success' => true, + 'provider' => $provider, + 'message' => '', + ); + + $helper->load_shipping_providers(); + + if ( $shipping_provider = $helper->get_shipping_provider( $provider ) ) { + $shipping_provider->delete(); + wp_send_json( $response ); + } else { + wp_send_json( $response_error ); + } + } + + public static function sort_shipping_provider() { + check_ajax_referer( 'sort-shipping-provider', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['order'] ) ) { + wp_die( -1 ); + } + + $order = wc_clean( wp_unslash( $_POST['order'] ) ); + $order_count = 0; + $helper = Helper::instance(); + $response = array( + 'success' => true, + 'message' => '', + ); + + $helper->load_shipping_providers(); + + foreach ( $order as $shipping_provider_name ) { + if ( $shipping_provider = $helper->get_shipping_provider( $shipping_provider_name ) ) { + $shipping_provider->set_order( ++$order_count ); + $shipping_provider->save(); + } + } + + wp_send_json( $response ); + } + + public static function shipments_bulk_action_handle() { + $action = isset( $_POST['bulk_action'] ) ? wc_clean( wp_unslash( $_POST['bulk_action'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Missing + $type = isset( $_POST['type'] ) ? wc_clean( wp_unslash( $_POST['type'] ) ) : 'simple'; // phpcs:ignore WordPress.Security.NonceVerification.Missing + + check_ajax_referer( "woocommerce_gzd_shipments_{$action}", 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['step'] ) || ! isset( $_POST['ids'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error while bulk processing shipments.', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + ); + + $handlers = Admin::get_bulk_action_handlers(); + + if ( ! array_key_exists( $action, $handlers ) ) { + wp_send_json( $response_error ); + } + + $ids = isset( $_POST['ids'] ) ? array_map( 'absint', $_POST['ids'] ) : array(); + $step = isset( $_POST['step'] ) ? absint( $_POST['step'] ) : 1; + + $handler = $handlers[ $action ]; + + if ( 1 === $step ) { + $handler->reset( true ); + } + + $handler->set_step( $step ); + $handler->set_ids( $ids ); + $handler->set_shipment_type( $type ); + + $handler->handle(); + + if ( $handler->get_percent_complete() >= 100 ) { + $errors = $handler->get_notices( 'error' ); + + if ( empty( $errors ) ) { + $handler->add_notice( $handler->get_success_message(), 'success' ); + $handler->update_notices(); + } + + wp_send_json_success( + array( + 'step' => 'done', + 'percentage' => 100, + 'url' => $handler->get_success_redirect_url(), + 'type' => $handler->get_shipment_type(), + ) + ); + } else { + wp_send_json_success( + array( + 'step' => ++$step, + 'percentage' => $handler->get_percent_complete(), + 'ids' => $handler->get_ids(), + 'type' => $handler->get_shipment_type(), + ) + ); + } + } + + /** + * @param Order $order + */ + private static function refresh_shipments( &$order ) { + MetaBox::refresh_shipments( $order ); + } + + /** + * @param Order $order + * @param bool $shipment + */ + private static function refresh_shipment_items( &$order, &$shipment = false ) { + MetaBox::refresh_shipment_items( $order, $shipment ); + } + + /** + * @param Order $order + */ + private static function refresh_status( &$order ) { + MetaBox::refresh_status( $order ); + } + + public static function update_shipment_status() { + if ( current_user_can( 'edit_shop_orders' ) && check_admin_referer( 'update-shipment-status' ) && isset( $_GET['status'], $_GET['shipment_id'] ) ) { + $status = sanitize_text_field( wp_unslash( $_GET['status'] ) ); + $shipment = wc_gzd_get_shipment( absint( wp_unslash( $_GET['shipment_id'] ) ) ); + + if ( wc_gzd_is_shipment_status( 'gzd-' . $status ) && $shipment ) { + $shipment->update_status( $status, true ); + /** + * Action to indicate Shipment status change via WP Admin. + * + * @param integer $shipment_id The shipment id. + * @param string $status The status to be switched to. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_updated_shipment_status', $shipment->get_id(), $status ); + } + } + + wp_safe_redirect( wp_get_referer() ? wp_get_referer() : admin_url( 'admin.php?page=wc-gzd-shipments' ) ); + exit; + } + + private static function get_shipment_ids( $shipments ) { + return array_values( + array_map( + function( $s ) { + return $s->get_id(); + }, + $shipments + ) + ); + } + + public static function remove_shipment() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $shipment->get_order() ) ) { + wp_send_json( $response_error ); + } + + $shipment_ids = self::get_shipment_ids( $order_shipment->get_shipments() ); + + if ( $shipment->delete( true ) ) { + $order_shipment->remove_shipment( $shipment_id ); + + if ( 'return' === $shipment->get_type() ) { + $order_shipment->validate_shipments(); + } + + /* + * Check which shipments have been deleted (e.g. multiple in case a return has been removed) + */ + $shipments_removed = array_values( array_diff( $shipment_ids, self::get_shipment_ids( $order_shipment->get_shipments() ) ) ); + $response['shipment_id'] = $shipments_removed; + + $response['fragments'] = array( + '.order-shipping-status' => self::get_order_status_html( $order_shipment ), + '.order-return-status' => self::get_order_return_status_html( $order_shipment ), + ); + + self::send_json_success( $response, $order_shipment ); + } else { + wp_send_json( $response_error ); + } + } + + public static function add_shipment() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['order_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error while adding the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + ); + + $order_id = absint( $_POST['order_id'] ); + + if ( ! $order = wc_get_order( $order_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + wp_send_json( $response_error ); + } + + self::refresh_shipment_items( $order_shipment ); + + if ( ! $order_shipment->needs_shipping() ) { + $response_error['message'] = _x( 'This order contains enough shipments already.', 'shipments', 'woocommerce-germanized' ); + wp_send_json( $response_error ); + } + + $shipment = wc_gzd_create_shipment( $order_shipment ); + + if ( is_wp_error( $shipment ) ) { + wp_send_json( $response_error ); + } + + $order_shipment->add_shipment( $shipment ); + + // Mark as active + $is_active = true; + + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-shipment.php'; + $html = ob_get_clean(); + + $response['new_shipment'] = $html; + $response['new_shipment_type'] = $shipment->get_type(); + $response['fragments'] = array( + '.order-shipping-status' => self::get_order_status_html( $order_shipment ), + '.order-return-status' => self::get_order_return_status_html( $order_shipment ), + ); + + self::send_json_success( $response, $order_shipment ); + } + + public static function add_return_shipment() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['order_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + 'new_shipment' => '', + ); + + $order_id = absint( $_POST['order_id'] ); + $items = isset( $_POST['return_item'] ) ? (array) wc_clean( wp_unslash( $_POST['return_item'] ) ) : array(); + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) { + wp_send_json( $response_error ); + } + + self::refresh_shipment_items( $order_shipment ); + + if ( ! $order_shipment->needs_return() ) { + $response_error['message'] = _x( 'This order contains enough returns already.', 'shipments', 'woocommerce-germanized' ); + wp_send_json( $response_error ); + } + + $shipment = wc_gzd_create_return_shipment( $order_shipment, array( 'items' => $items ) ); + + if ( is_wp_error( $shipment ) ) { + wp_send_json( $response_error ); + } + + $order_shipment->add_shipment( $shipment ); + + // Mark as active + $is_active = true; + + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-shipment.php'; + $html = ob_get_clean(); + + $response['new_shipment'] = $html; + $response['new_shipment_type'] = $shipment->get_type(); + $response['fragments'] = array( + '.order-shipping-status' => self::get_order_status_html( $order_shipment ), + '.order-return-status' => self::get_order_return_status_html( $order_shipment ), + ); + + self::send_json_success( $response, $order_shipment ); + } + + public static function validate_shipment_item_quantities() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['order_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + ); + + $order_id = absint( $_POST['order_id'] ); + $active = isset( $_POST['active'] ) ? absint( $_POST['active'] ) : 0; + + if ( ! $order = wc_get_order( $order_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + wp_send_json( $response_error ); + } + + static::refresh_shipments( $order_shipment ); + + $order_shipment->validate_shipments(); + + $response['fragments'] = self::get_shipments_html( $order_shipment, $active ); + + self::send_json_success( $response, $order_shipment ); + } + + public static function sync_shipment_items() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order = $shipment->get_order() ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + wp_send_json( $response_error ); + } + + $shipment = $order_shipment->get_shipment( $shipment_id ); + + static::refresh_shipment_items( $order_shipment ); + + if ( $shipment->is_editable() ) { + $shipment = $order_shipment->get_shipment( $shipment_id ); + + // Make sure we are working based on the current instance. + $shipment->set_order_shipment( $order_shipment ); + $shipment->sync_items(); + $shipment->save(); + } + + ob_start(); + foreach ( $shipment->get_items() as $item ) { + include Package::get_path() . '/includes/admin/views/html-order-shipment-item.php'; + } + $html = ob_get_clean(); + + $response['fragments'] = array( + '#shipment-' . $shipment->get_id() . ' .shipment-item-list:first' => '
' . $html . '
', + '#shipment-' . $shipment->get_id() . ' .item-count:first' => self::get_item_count_html( $shipment, $order_shipment ), + ); + + self::send_json_success( $response, $order_shipment, $shipment ); + } + + public static function json_search_shipping_provider() { + ob_start(); + + check_ajax_referer( 'search-shipping-provider', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) ) { + wp_die( -1 ); + } + + $term = isset( $_GET['term'] ) ? (string) wc_clean( wp_unslash( $_GET['term'] ) ) : ''; + $found_providers = array(); + + if ( empty( $term ) ) { + wp_die(); + } + + global $wpdb; + + $names = $wpdb->get_col( + $wpdb->prepare( + "SELECT DISTINCT p1.shipping_provider_name FROM {$wpdb->gzd_shipping_provider} p1 WHERE p1.shipping_provider_title LIKE %s AND p1.shipping_provider_activated = 1", // @codingStandardsIgnoreLine + $wpdb->esc_like( wc_clean( $term ) ) . '%' + ) + ); + + foreach ( $names as $name ) { + if ( $shipping_provider = wc_gzd_get_shipping_provider( $name ) ) { + $found_providers[ $name ] = esc_html( $shipping_provider->get_title() ); + } + } + + /** + * Filter to adjust found shipping providers to filter Shipments. + * + * @param array $result The shipping provider search result. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + wp_send_json( apply_filters( 'woocommerce_gzd_json_search_found_shipment_shipping_providers', $found_providers ) ); + } + + public static function json_search_orders() { + ob_start(); + + check_ajax_referer( 'search-orders', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) ) { + wp_die( -1 ); + } + + $term = isset( $_GET['term'] ) ? (string) wc_clean( wp_unslash( $_GET['term'] ) ) : ''; + $limit = 0; + + if ( empty( $term ) ) { + wp_die(); + } + + if ( Package::is_hpos_enabled() ) { + $ids = wc_get_orders( array( 's' => $term ) ); + } else { + if ( ! is_numeric( $term ) ) { + $ids = wc_get_orders( array( 's' => $term ) ); + } else { + global $wpdb; + + $ids = $wpdb->get_col( + $wpdb->prepare( + "SELECT DISTINCT p1.ID FROM {$wpdb->posts} p1 WHERE p1.ID LIKE %s AND post_type = 'shop_order'", // @codingStandardsIgnoreLine + $wpdb->esc_like( wc_clean( $term ) ) . '%' + ) + ); + } + } + + $excluded = array(); + + if ( ! empty( $_GET['exclude'] ) ) { + $excluded = array_map( 'absint', (array) wp_unslash( $_GET['exclude'] ) ); + } + + foreach ( $ids as $id ) { + if ( $order = wc_get_order( $id ) ) { + if ( in_array( absint( $order->get_id() ), $excluded, true ) ) { + continue; + } + + $found_orders[ $order->get_id() ] = sprintf( + esc_html_x( 'Order #%s', 'shipments', 'woocommerce-germanized' ), + $order->get_order_number() + ); + } + } + + /** + * Filter to adjust found orders to filter Shipments. + * + * @param array $result The order search result. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + wp_send_json( apply_filters( 'woocommerce_gzd_json_search_found_shipment_orders', $found_orders ) ); + } + + private static function get_order_status_html( $order_shipment ) { + $status_html = '' . wc_gzd_get_shipment_order_shipping_status_name( $order_shipment->get_shipping_status() ) . ''; + + return $status_html; + } + + /** + * @param \WC_Order $order + * + * @return string[] + */ + private static function get_global_order_status_html( $order ) { + $old_status = $order->get_status(); + $result = array( + 'status' => '', + 'input' => '', + ); + + /** + * Load a clean instance to make sure order status updates are reflected. + */ + $order = wc_get_order( $order->get_id() ); + + /** + * In case the current request has not changed the status do not return html + */ + if ( ! $order || $old_status === $order->get_status() ) { + return $result; + } + ob_start(); + ?> +

+ + +

+ get_status( 'edit' ) ) . '" />'; + + return $result; + } + + private static function get_order_return_status_html( $order_shipment ) { + $status_html = '' . wc_gzd_get_shipment_order_return_status_name( $order_shipment->get_return_status() ) . ''; + + return $status_html; + } + + public static function refresh_shipment_packaging() { + check_ajax_referer( 'refresh-shipment-packaging', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + $response = array( + 'success' => true, + 'message' => '', + 'shipment_id' => $shipment_id, + 'needs_packaging_refresh' => true, + ); + + $data = array( + 'packaging_id' => isset( $_POST['shipment_packaging_id'][ $shipment_id ] ) ? absint( wc_clean( wp_unslash( $_POST['shipment_packaging_id'][ $shipment_id ] ) ) ) : '', + ); + + $shipment->set_props( $data ); + + $response['fragments'] = array( + '#shipment-' . $shipment->get_id() . ' .shipment-packaging-select' => self::get_packaging_select_html( $shipment ), + ); + + wp_send_json( $response ); + } + + protected static function get_packaging_select_html( $shipment ) { + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-shipment-packaging-select.php'; + $html = ob_get_clean(); + + return $html; + } + + public static function save_shipments() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['order_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + ); + + $order_id = absint( $_POST['order_id'] ); + $active = isset( $_POST['active'] ) ? absint( $_POST['active'] ) : 0; + + if ( ! $order = wc_get_order( $order_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + wp_send_json( $response_error ); + } + + // Refresh data + self::refresh_shipments( $order_shipment ); + + // Make sure that we are not applying more + $order_shipment->validate_shipment_item_quantities(); + + // Refresh statuses after adjusting quantities + self::refresh_status( $order_shipment ); + + $order_shipment->save(); + + $response['fragments'] = self::get_shipments_html( $order_shipment, $active ); + + self::send_json_success( $response, $order_shipment ); + } + + /** + * @param Order $order_shipment + * @param int $active + * + * @return array + */ + private static function get_shipments_html( $p_order_shipment, $p_active = 0 ) { + $order_shipment = $p_order_shipment; + $active_shipment = $p_active; + + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-shipment-list.php'; + $html = ob_get_clean(); + + $order_status_html = self::get_global_order_status_html( $order_shipment->get_order() ); + + $fragments = array( + '#order-shipments-list' => $html, + '.order-shipping-status' => self::get_order_status_html( $order_shipment ), + '.order-return-status' => self::get_order_return_status_html( $order_shipment ), + '.order_data_column p.wc-order-status' => $order_status_html['status'], + 'input[name="post_status"]' => $order_status_html['input'], + ); + + if ( empty( $fragments['.order_data_column p.wc-order-status'] ) ) { + unset( $fragments['.order_data_column p.wc-order-status'] ); + unset( $fragments['input[name="post_status"]'] ); + } + + return $fragments; + } + + public static function get_available_return_shipment_items() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['order_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + 'html' => '', + ); + + $order_id = absint( $_POST['order_id'] ); + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) { + wp_send_json( $response_error ); + } + + static::refresh_shipments( $order_shipment ); + + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-add-return-shipment-items.php'; + $response['html'] = ob_get_clean(); + + self::send_json_success( $response, $order_shipment ); + } + + public static function get_available_shipment_items() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + 'items' => array(), + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order = $shipment->get_order() ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + wp_send_json( $response_error ); + } + + static::refresh_shipments( $order_shipment ); + + if ( 'return' === $shipment->get_type() ) { + $response['items'] = $order_shipment->get_available_items_for_return( + array( + 'shipment_id' => $shipment->get_id(), + 'disable_duplicates' => true, + ) + ); + } else { + $response['items'] = $order_shipment->get_available_items_for_shipment( + array( + 'shipment_id' => $shipment_id, + 'disable_duplicates' => true, + ) + ); + } + + self::send_json_success( $response, $order_shipment ); + } + + public static function add_shipment_item() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + 'new_item' => '', + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + $original_item_id = isset( $_POST['original_item_id'] ) ? absint( $_POST['original_item_id'] ) : 0; + $item_quantity = isset( $_POST['quantity'] ) ? absint( $_POST['quantity'] ) : false; + + if ( false !== $item_quantity && 0 === $item_quantity ) { + $item_quantity = 1; + } + + if ( empty( $original_item_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order = $shipment->get_order() ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + wp_send_json( $response_error ); + } + + static::refresh_shipments( $order_shipment ); + + // Make sure we are working with the shipment from the order + $shipment = $order_shipment->get_shipment( $shipment_id ); + + if ( 'return' === $shipment->get_type() ) { + $item = self::add_shipment_return_item( $order_shipment, $shipment, $original_item_id, $item_quantity ); + } else { + $item = self::add_shipment_order_item( $order_shipment, $shipment, $original_item_id, $item_quantity ); + } + + if ( ! $item ) { + wp_send_json( $response_error ); + } + + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-shipment-item.php'; + $response['new_item'] = ob_get_clean(); + + $response['fragments'] = array( + '#shipment-' . $shipment->get_id() . ' .item-count:first' => self::get_item_count_html( $shipment, $order_shipment ), + ); + + self::send_json_success( $response, $order_shipment, $shipment ); + } + + /** + * @param Order $order_shipment + * @param ReturnShipment $shipment + * @param integer $parent_item_id + * @param integer $quantity + */ + private static function add_shipment_return_item( $order_shipment, $shipment, $order_item_id, $quantity ) { + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + if ( ! $shipment_item = $order_shipment->get_simple_shipment_item( $order_item_id ) ) { + wp_send_json( $response_error ); + } + + // No duplicates allowed + if ( $shipment->get_item_by_order_item_id( $order_item_id ) ) { + wp_send_json( $response_error ); + } + + // Check max quantity + $quantity_left = $order_shipment->get_item_quantity_left_for_returning( $shipment_item->get_order_item_id() ); + + if ( $quantity ) { + if ( $quantity > $quantity_left ) { + $quantity = $quantity_left; + } + } else { + $quantity = $quantity_left; + } + + if ( $item = wc_gzd_create_return_shipment_item( $shipment, $shipment_item, array( 'quantity' => $quantity ) ) ) { + $shipment->add_item( $item ); + $shipment->save(); + } + + return $item; + } + + /** + * @param Order $order_shipment + * @param SimpleShipment $shipment + * @param integer $order_item_id + * @param integer $quantity + */ + private static function add_shipment_order_item( $order_shipment, $shipment, $order_item_id, $quantity ) { + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $order = $order_shipment->get_order(); + + if ( ! $order_item = $order->get_item( $order_item_id ) ) { + wp_send_json( $response_error ); + } + + // No duplicates allowed + if ( $shipment->get_item_by_order_item_id( $order_item_id ) ) { + wp_send_json( $response_error ); + } + + // Check max quantity + $quantity_left = $order_shipment->get_item_quantity_left_for_shipping( $order_item ); + + if ( $quantity ) { + if ( $quantity > $quantity_left ) { + $quantity = $quantity_left; + } + } else { + $quantity = $quantity_left; + } + + if ( $item = wc_gzd_create_shipment_item( $shipment, $order_item, array( 'quantity' => $quantity ) ) ) { + $shipment->add_item( $item ); + $shipment->save(); + } + + return $item; + } + + private static function get_item_count_html( $p_shipment, $p_order_shipment ) { + $shipment = $p_shipment; + + // Refresh the instance to make sure we are working with the same object + $shipment->set_order_shipment( $p_order_shipment ); + + ob_start(); + include Package::get_path() . '/includes/admin/views/html-order-shipment-item-count.php'; + $html = ob_get_clean(); + + return $html; + } + + public static function remove_shipment_item() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) || ! isset( $_POST['item_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + 'item_id' => '', + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + $item_id = absint( $_POST['item_id'] ); + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $item = $shipment->get_item( $item_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $shipment->get_order_id() ) ) { + wp_send_json( $response_error ); + } + + $shipment->remove_item( $item_id ); + $shipment->save(); + + $response['item_id'] = $item_id; + $response['fragments'] = array( + '#shipment-' . $shipment->get_id() . ' .item-count:first' => self::get_item_count_html( $shipment, $order_shipment ), + ); + + self::send_json_success( $response, $order_shipment, $shipment ); + } + + public static function limit_shipment_item_quantity() { + check_ajax_referer( 'edit-shipments', 'security' ); + + if ( ! current_user_can( 'edit_shop_orders' ) || ! isset( $_POST['shipment_id'] ) || ! isset( $_POST['item_id'] ) ) { + wp_die( -1 ); + } + + $response_error = array( + 'success' => false, + 'message' => _x( 'There was an error processing the shipment', 'shipments', 'woocommerce-germanized' ), + ); + + $response = array( + 'success' => true, + 'message' => '', + 'max_quantity' => '', + 'item_id' => '', + ); + + $shipment_id = absint( $_POST['shipment_id'] ); + $item_id = absint( $_POST['item_id'] ); + $quantity = isset( $_POST['quantity'] ) ? (int) $_POST['quantity'] : 1; + $quantity = $quantity <= 0 ? 1 : $quantity; + + if ( ! $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order = $shipment->get_order() ) { + wp_send_json( $response_error ); + } + + if ( ! $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + wp_send_json( $response_error ); + } + + // Make sure the shipment order gets notified about changes + if ( ! $shipment = $order_shipment->get_shipment( $shipment_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $item = $shipment->get_item( $item_id ) ) { + wp_send_json( $response_error ); + } + + if ( ! $order_item = $order->get_item( $item->get_order_item_id() ) ) { + wp_send_json( $response_error ); + } + + static::refresh_shipments( $order_shipment ); + + $quantity_max = 0; + + if ( 'return' === $shipment->get_type() ) { + $quantity_max = $order_shipment->get_item_quantity_left_for_returning( + $item->get_order_item_id(), + array( + 'exclude_current_shipment' => true, + 'shipment_id' => $shipment->get_id(), + ) + ); + } else { + $quantity_max = $order_shipment->get_item_quantity_left_for_shipping( + $order_item, + array( + 'exclude_current_shipment' => true, + 'shipment_id' => $shipment->get_id(), + ) + ); + } + + $response['item_id'] = $item_id; + $response['max_quantity'] = $quantity_max; + + if ( $quantity > $quantity_max ) { + $quantity = $quantity_max; + } + + $shipment->update_item_quantity( $item_id, $quantity ); + + $response['fragments'] = array( + '#shipment-' . $shipment->get_id() . ' .item-count:first' => self::get_item_count_html( $shipment, $order_shipment ), + ); + + self::send_json_success( $response, $order_shipment, $shipment ); + } + + /** + * @param $response + * @param Order $order_shipment + * @param Shipment|bool $shipment + */ + private static function send_json_success( $response, $order_shipment, $current_shipment = false ) { + + $available_items = $order_shipment->get_available_items_for_shipment(); + $response['shipments'] = array(); + + foreach ( $order_shipment->get_shipments() as $shipment ) { + $shipment->set_order_shipment( $order_shipment ); + + $response['shipments'][ $shipment->get_id() ] = array( + 'is_editable' => $shipment->is_editable(), + 'needs_items' => $shipment->needs_items( array_keys( $available_items ) ), + 'weight' => wc_format_localized_decimal( $shipment->get_content_weight() ), + 'length' => wc_format_localized_decimal( $shipment->get_content_length() ), + 'width' => wc_format_localized_decimal( $shipment->get_content_width() ), + 'height' => wc_format_localized_decimal( $shipment->get_content_height() ), + 'total_weight' => wc_format_localized_decimal( $shipment->get_total_weight() ), + ); + } + + $response['order_needs_new_shipments'] = $order_shipment->needs_shipping(); + $response['order_needs_new_returns'] = $order_shipment->needs_return(); + + if ( $current_shipment ) { + if ( ! isset( $response['fragments'] ) ) { + $response['fragments'] = array(); + } + + $response['needs_packaging_refresh'] = true; + $response['shipment_id'] = $current_shipment->get_id(); + $response['fragments'][ '#shipment-' . $current_shipment->get_id() . ' .shipment-packaging-select' ] = self::get_packaging_select_html( $current_shipment ); + } + + wp_send_json( $response ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Api.php b/packages/woocommerce-germanized-shipments/src/Api.php new file mode 100644 index 000000000..51cc8d7c8 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Api.php @@ -0,0 +1,215 @@ +set_data( array_merge( $response->data, self::get_product_data( $product, $context ) ) ); + } + + return $response; + } + + private static function get_product_data( $product, $context = 'view' ) { + $data = array(); + + if ( $shipments_product = wc_gzd_shipments_get_product( $product ) ) { + $data['hs_code'] = $shipments_product->get_hs_code( $context ); + $data['manufacture_country'] = $shipments_product->get_manufacture_country( $context ); + } + + return $data; + } + + /** + * @param \WC_Product $product + * @param $request + * + * @return \WC_Product $product + */ + public static function update_product( $product, $request ) { + if ( $shipments_product = wc_gzd_shipments_get_product( $product ) ) { + if ( isset( $request['hs_code'] ) ) { + $shipments_product->set_hs_code( wc_clean( wp_unslash( $request['hs_code'] ) ) ); + } + + if ( isset( $request['manufacture_country'] ) ) { + $shipments_product->set_manufacture_country( wc_clean( wp_unslash( $request['manufacture_country'] ) ) ); + } + } + + return $product; + } + + public static function remove_status_prefix( $status ) { + if ( 'gzd-' === substr( $status, 0, 4 ) ) { + $status = substr( $status, 4 ); + } + + return $status; + } + + /** + * Extend schema. + * + * @since 1.0.0 + * + * @param array $schema_properties Data used to create the order. + * + * @return array + */ + public static function order_schema( $schema_properties ) { + $statuses = array_map( array( __CLASS__, 'remove_status_prefix' ), array_keys( wc_gzd_get_shipment_order_shipping_statuses() ) ); + + $schema_properties['shipping_status'] = array( + 'description' => _x( 'Shipping status', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'enum' => $statuses, + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ); + + return $schema_properties; + } + + /** + * Extend product variation schema. + * + * @since 1.0.0 + * + * @param array $schema_properties Data used to create the product. + * + * @return array + */ + public static function product_variation_schema( $schema_properties ) { + $schema_properties['hs_code'] = array( + 'description' => _x( 'HS-Code (Customs)', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ); + + $schema_properties['manufacture_country'] = array( + 'description' => _x( 'Country of manufacture (Customs)', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ); + + return $schema_properties; + } + + /** + * Extend product schema. + * + * @since 1.0.0 + * + * @param array $schema_properties Data used to create the product. + * + * @return array + */ + public static function product_schema( $schema_properties ) { + $schema_properties['hs_code'] = array( + 'description' => _x( 'HS-Code (Customs)', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ); + + $schema_properties['manufacture_country'] = array( + 'description' => _x( 'Country of manufacture (Customs)', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ); + + return $schema_properties; + } + + /** + * @param WP_REST_Response $response + * @param $post + * @param WP_REST_Request $request + * + * @return mixed + */ + public static function prepare_order_shipments( $response, $post, $request ) { + $order = wc_get_order( $post ); + $response_order_data = $response->get_data(); + $response_order_data['shipments'] = array(); + $response_order_data['shipping_status'] = 'no-shipping-needed'; + + if ( $order ) { + $order_shipment = wc_gzd_get_shipment_order( $order ); + $shipments = $order_shipment->get_shipments(); + + if ( ! empty( $shipments ) ) { + foreach ( $shipments as $shipment ) { + $response_order_data['shipments'][] = ShipmentsController::prepare_shipment( $shipment, 'view', $request['dp'] ); + } + } + + $response_order_data['shipping_status'] = $order_shipment->get_shipping_status(); + } + + $response->set_data( $response_order_data ); + + return $response; + } + + public static function order_shipments_schema( $schema ) { + $schema['shipments'] = array( + 'description' => _x( 'List of shipments.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + 'items' => ShipmentsController::get_single_item_schema(), + ); + + return $schema; + } + + protected static function get_shipment_statuses() { + $statuses = array(); + + foreach ( array_keys( wc_gzd_get_shipment_statuses() ) as $status ) { + $statuses[] = str_replace( 'gzd-', '', $status ); + } + + return $statuses; + } + + public static function register_controllers( $controller ) { + $controller['wc/v3']['shipments'] = ShipmentsController::class; + + return $controller; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Automation.php b/packages/woocommerce-germanized-shipments/src/Automation.php new file mode 100644 index 000000000..fcca035e8 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Automation.php @@ -0,0 +1,257 @@ +get_id() ) { + self::maybe_create_shipments( $order ); + + remove_action( 'woocommerce_after_order_object_save', array( __CLASS__, 'after_new_order' ), 150 ); + self::$current_new_order_id = null; + } + } + + /** + * @param $order_id + * @param $old_status + * @param $new_status + * @param WC_Order $order + */ + public static function maybe_mark_shipments_shipped( $order_id, $old_status, $new_status, $order ) { + + /** + * Filter to decide which order status is used to determine if a order + * is completed or not to update contained shipment statuses to shipped. + * Does only take effect if the automation option has been set within the shipment settings. + * + * @param string $status The current order status. + * @param integer $order_id The order id. + * + * @since 3.0.5 + * @package Vendidero/Germanized/Shipments + */ + if ( apply_filters( 'woocommerce_gzd_shipments_order_completed_status', 'completed', $order_id ) === $new_status ) { + + // Make sure that MetaBox is saved before we process automation + if ( self::is_admin_edit_order_request() ) { + add_action( 'woocommerce_process_shop_order_meta', array( __CLASS__, 'mark_shipments_shipped' ), 70 ); + } else { + self::mark_shipments_shipped( $order_id ); + } + } + } + + private static function is_admin_edit_order_request() { + return ( isset( $_POST['action'] ) && ( ( 'editpost' === $_POST['action'] && isset( $_POST['post_type'] ) && 'shop_order' === $_POST['post_type'] ) || 'edit_order' === $_POST['action'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + public static function mark_shipments_shipped( $order_id ) { + if ( $order = wc_get_order( $order_id ) ) { + if ( $shipment_order = wc_gzd_get_shipment_order( $order ) ) { + foreach ( $shipment_order->get_simple_shipments() as $shipment ) { + + if ( ! $shipment->is_shipped() ) { + $shipment->update_status( 'shipped' ); + } + } + } + } + } + + /** + * Mark the order as completed if the order is fully shipped. + * + * @param $order_id + */ + public static function mark_order_completed( $order_id ) { + if ( $order = wc_get_order( $order_id ) ) { + + /** + * By default do not mark orders (via invoice) as completed after shipped as + * the order will be shipped before the invoice was paid. + */ + $mark_as_completed = ! in_array( $order->get_payment_method(), array( 'invoice' ), true ) ? true : false; + + /** + * Filter that allows to conditionally disable automatic + * order completion after the shipments are marked as shipped. + * + * @param boolean $mark_as_completed Whether to mark the order as completed or not. + * @param integer $order_id The order id. + * + * @since 3.2.3 + * @package Vendidero/Germanized/Shipments + */ + if ( ! apply_filters( 'woocommerce_gzd_shipment_order_mark_as_completed', $mark_as_completed, $order_id ) ) { + return; + } + + /** + * Filter to adjust the new status of an order after all it's required + * shipments have been marked as shipped. Does only take effect if the automation option has been set + * within the shipment settings. + * + * @param string $status The order status to be used. + * @param integer $order_id The order id. + * + * @since 3.0.5 + * @package Vendidero/Germanized/Shipments + */ + $order->update_status( apply_filters( 'woocommerce_gzd_shipment_order_completed_status', 'completed', $order_id ), _x( 'Order is fully shipped.', 'shipments', 'woocommerce-germanized' ) ); + } + } + + public static function create_shipments( $order, $enable_auto_filter = true ) { + if ( is_numeric( $order ) ) { + $order = wc_get_order( $order ); + } + + if ( ! $order ) { + return; + } + + $shipment_status = Package::get_setting( 'auto_default_status' ); + + if ( empty( $shipment_status ) ) { + $shipment_status = 'processing'; + } + + /** + * Filter to disable automatically creating shipments for a specific order. + * + * @param string $enable Whether to create or not create shipments. + * @param integer $order_id The order id. + * @param WC_Order $order The order instance. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + if ( $enable_auto_filter && ! apply_filters( 'woocommerce_gzd_auto_create_shipments_for_order', true, $order->get_id(), $order ) ) { + return; + } + + if ( $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + if ( ! apply_filters( 'woocommerce_gzd_auto_create_custom_shipments_for_order', false, $order->get_id(), $order ) ) { + $shipments = $order_shipment->get_simple_shipments(); + + foreach ( $shipments as $shipment ) { + if ( $shipment->is_editable() ) { + $shipment->sync(); + $shipment->sync_items(); + $shipment->save(); + } + } + + if ( $order_shipment->needs_shipping() ) { + $shipment = wc_gzd_create_shipment( $order_shipment, array( 'props' => array( 'status' => $shipment_status ) ) ); + + if ( ! is_wp_error( $shipment ) ) { + $order_shipment->add_shipment( $shipment ); + } + } + } + + do_action( 'woocommerce_gzd_after_auto_create_shipments_for_order', $order->get_id(), $shipment_status, $order ); + } + } + + protected static function get_auto_statuses() { + $statuses = (array) Package::get_setting( 'auto_statuses' ); + $clean_statuses = array(); + + if ( ! empty( $statuses ) ) { + foreach ( $statuses as $status ) { + $status = trim( str_replace( 'wc-', '', $status ) ); + + if ( ! in_array( $status, $clean_statuses, true ) ) { + $clean_statuses[] = $status; + } + } + } + + return $clean_statuses; + } + + public static function maybe_create_shipments( $order_id ) { + $statuses = self::get_auto_statuses(); + $has_status = empty( $statuses ) ? true : false; + + if ( ! $has_status ) { + if ( $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) { + $has_status = $order_shipment->get_order()->has_status( $statuses ); + } + } + + if ( $has_status ) { + // Make sure that MetaBox is saved before we process automation + if ( self::is_admin_edit_order_request() ) { + add_action( 'woocommerce_process_shop_order_meta', array( __CLASS__, 'create_shipments' ), 70 ); + } else { + self::create_shipments( $order_id ); + } + } + } + + public static function maybe_create_subscription_shipments( $renewal_order ) { + self::create_shipments( $renewal_order->get_id() ); + + return $renewal_order; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/DataStores/Label.php b/packages/woocommerce-germanized-shipments/src/DataStores/Label.php new file mode 100644 index 000000000..f468644d3 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/DataStores/Label.php @@ -0,0 +1,557 @@ +set_date_created( time() ); + + $data = array( + 'label_number' => $label->get_number(), + 'label_shipment_id' => $label->get_shipment_id(), + 'label_path' => $label->get_path(), + 'label_product_id' => $label->get_product_id(), + 'label_type' => $label->get_type(), + 'label_shipping_provider' => $label->get_shipping_provider(), + 'label_parent_id' => $label->get_parent_id(), + 'label_date_created' => gmdate( 'Y-m-d H:i:s', $label->get_date_created( 'edit' )->getOffsetTimestamp() ), + 'label_date_created_gmt' => gmdate( 'Y-m-d H:i:s', $label->get_date_created( 'edit' )->getTimestamp() ), + ); + + $wpdb->insert( + $wpdb->gzd_shipment_labels, + $data + ); + + $label_id = $wpdb->insert_id; + + if ( $label_id ) { + $label->set_id( $label_id ); + + $this->save_label_data( $label ); + + $label->save_meta_data(); + $label->apply_changes(); + + $this->clear_caches( $label ); + + $hook_postfix = $this->get_hook_postfix( $label ); + + /** + * Action fires when a new DHL label has been created. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * label type e.g. return in case it is not a simple label. + * + * @param integer $label_id The label id. + * @param Label $label The label instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + do_action( "woocommerce_gzd_shipment_{$hook_postfix}label_created", $label_id, $label ); + } + } + + /** + * @param \Vendidero\Germanized\Shipments\Labels\Label $label + * + * @return string + */ + protected function get_hook_postfix( $label ) { + $prefix = $label->get_shipping_provider() . '_'; + + if ( 'simple' !== $label->get_type() ) { + $prefix = $prefix . $label->get_type() . '_'; + } + + return $prefix; + } + + /** + * Method to update a label in the database. + * + * @param \Vendidero\Germanized\Shipments\Labels\Label $label Label object. + */ + public function update( &$label ) { + global $wpdb; + + $updated_props = array(); + $core_props = $this->core_props; + $changed_props = array_keys( $label->get_changes() ); + $label_data = array(); + + foreach ( $changed_props as $prop ) { + + if ( ! in_array( $prop, $core_props, true ) ) { + continue; + } + + switch ( $prop ) { + case 'date_created': + $label_data[ 'label' . $prop ] = gmdate( 'Y-m-d H:i:s', $label->{'get_' . $prop}( 'edit' )->getOffsetTimestamp() ); + $label_data[ 'label_' . $prop . '_gmt' ] = gmdate( 'Y-m-d H:i:s', $label->{'get_' . $prop}( 'edit' )->getTimestamp() ); + break; + default: + if ( is_callable( array( $label, 'get_' . $prop ) ) ) { + $label_data[ 'label_' . $prop ] = $label->{'get_' . $prop}( 'edit' ); + } + break; + } + } + + if ( ! empty( $label_data ) ) { + $wpdb->update( + $wpdb->gzd_shipment_labels, + $label_data, + array( 'label_id' => $label->get_id() ) + ); + } + + $this->save_label_data( $label ); + $label->save_meta_data(); + + $label->apply_changes(); + $this->clear_caches( $label ); + + $hook_postfix = $this->get_hook_postfix( $label ); + + /** + * Action fires after a DHL label has been updated in the DB. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * label type e.g. return in case it is not a simple label. + * + * @param integer $label_id The label id. + * @param Label $label The label instance. + * @param array $changed_props Properties that have been changed. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + do_action( "woocommerce_gzd_shipment_{$hook_postfix}label_updated", $label->get_id(), $label, $changed_props ); + } + + /** + * Remove a shipment from the database. + * + * @since 3.0.0 + * @param \Vendidero\Germanized\Shipments\Labels\Label $label Label object. + * @param bool $force_delete Unused param. + */ + public function delete( &$label, $force_delete = false ) { + global $wpdb; + + if ( $file = $label->get_file() ) { + wp_delete_file( $file ); + } + + /* + * Delete additional files e.g. export documents + */ + foreach ( $label->get_additional_file_types() as $file_type ) { + if ( $file = $label->get_file( $file_type ) ) { + wp_delete_file( $file ); + } + } + + $wpdb->delete( $wpdb->gzd_shipment_labels, array( 'label_id' => $label->get_id() ), array( '%d' ) ); + $wpdb->delete( $wpdb->gzd_shipment_labelmeta, array( 'gzd_shipment_label_id' => $label->get_id() ), array( '%d' ) ); + + $this->clear_caches( $label ); + + $children = $label->get_children(); + + foreach ( $children as $child ) { + $child->delete( $force_delete ); + } + + $hook_postfix = $this->get_hook_postfix( $label ); + + /** + * Action fires after a DHL label has been deleted from DB. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * label type e.g. return in case it is not a simple label. + * + * @param integer $label_id The label id. + * @param \Vendidero\Germanized\DHL\Label $label The label object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + do_action( "woocommerce_gzd_shipment_{$hook_postfix}label_deleted", $label->get_id(), $label ); + } + + /** + * Read a shipment from the database. + * + * @since 3.0.0 + * + * @param \Vendidero\Germanized\Shipments\Labels\Label $label Label object. + * + * @throws Exception Throw exception if invalid shipment. + */ + public function read( &$label ) { + global $wpdb; + + $data = $wpdb->get_row( + $wpdb->prepare( + "SELECT * FROM {$wpdb->gzd_shipment_labels} WHERE label_id = %d LIMIT 1", + $label->get_id() + ) + ); + + if ( $data ) { + $label->set_props( + array( + 'shipment_id' => $data->label_shipment_id, + 'number' => $data->label_number, + 'path' => $data->label_path, + 'parent_id' => $data->label_parent_id, + 'shipping_provider' => $data->label_shipping_provider, + 'product_id' => $data->label_product_id, + 'date_created' => '0000-00-00 00:00:00' !== $data->label_date_created_gmt ? wc_string_to_timestamp( $data->label_date_created_gmt ) : null, + ) + ); + + $this->read_label_data( $label ); + $label->read_meta_data(); + $label->set_object_read( true ); + + $hook_postfix = $this->get_hook_postfix( $label ); + + /** + * Action fires after reading a DHL label from DB. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * label type e.g. return in case it is not a simple label. + * + * @param \Vendidero\Germanized\Shipments\Labels\Label $label The label object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + do_action( "woocommerce_gzd_shipment_{$hook_postfix}label_loaded", $label ); + } else { + throw new Exception( _x( 'Invalid label.', 'shipments', 'woocommerce-germanized' ) ); + } + } + + /** + * Clear any caches. + * + * @param \Vendidero\Germanized\Shipments\Labels\Label $label Label object. + * @since 3.0.0 + */ + protected function clear_caches( &$label ) { + wp_cache_delete( $label->get_id(), $this->meta_type . '_meta' ); + } + + /* + |-------------------------------------------------------------------------- + | Additional Methods + |-------------------------------------------------------------------------- + */ + + /** + * Get the label type based on label ID. + * + * @param int $label_id Label id. + * @return bool|mixed + */ + public function get_label_data( $label_id ) { + global $wpdb; + + $data = $wpdb->get_row( + $wpdb->prepare( + "SELECT label_type, label_shipping_provider FROM {$wpdb->gzd_shipment_labels} WHERE label_id = %d LIMIT 1", + $label_id + ) + ); + + if ( ! empty( $data ) ) { + return (object) array( + 'shipping_provider' => $data->label_shipping_provider, + 'type' => $data->label_type, + ); + } + + return false; + } + + /** + * Read extra data associated with the shipment. + * + * @param \Vendidero\Germanized\Shipments\Labels\Label $label Label object. + * @since 3.0.0 + */ + protected function read_label_data( &$label ) { + $props = array(); + $meta_keys = $this->internal_meta_keys; + + foreach ( $label->get_extra_data_keys() as $key ) { + $meta_keys[] = '_' . $key; + } + + foreach ( $meta_keys as $meta_key ) { + $props[ substr( $meta_key, 1 ) ] = get_metadata( $this->meta_type, $label->get_id(), $meta_key, true ); + } + + $label->set_props( $props ); + } + + /** + * @param \Vendidero\Germanized\Shipments\Labels\Label $label + */ + protected function save_label_data( &$label ) { + $updated_props = array(); + $meta_key_to_props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + $prop_name = substr( $meta_key, 1 ); + + if ( in_array( $prop_name, $this->core_props, true ) ) { + continue; + } + + $meta_key_to_props[ $meta_key ] = $prop_name; + } + + // Make sure to take extra data (like product url or text for external products) into account. + $extra_data_keys = $label->get_extra_data_keys(); + + foreach ( $extra_data_keys as $key ) { + $meta_key_to_props[ '_' . $key ] = $key; + } + + $props_to_update = $this->get_props_to_update( $label, $meta_key_to_props, $this->meta_type ); + + foreach ( $props_to_update as $meta_key => $prop ) { + + if ( ! is_callable( array( $label, "get_$prop" ) ) ) { + continue; + } + + $value = $label->{"get_$prop"}( 'edit' ); + $value = is_string( $value ) ? wp_slash( $value ) : $value; + + if ( is_bool( $value ) ) { + $value = wc_bool_to_string( $value ); + } + + $updated = $this->update_or_delete_meta( $label, $meta_key, $value ); + + if ( $updated ) { + $updated_props[] = $prop; + } + } + + /** + * Action fires after DHL label meta properties have been updated. + * + * @param \Vendidero\Germanized\Shipments\Labels\Label $label The label object. + * @param array $updated_props The updated properties. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + do_action( 'woocommerce_gzd_shipment_label_object_updated_props', $label, $updated_props ); + } + + /** + * Update meta data in, or delete it from, the database. + * + * Avoids storing meta when it's either an empty string or empty array. + * Other empty values such as numeric 0 and null should still be stored. + * Data-stores can force meta to exist using `must_exist_meta_keys`. + * + * Note: WordPress `get_metadata` function returns an empty string when meta data does not exist. + * + * @param WC_Data $object The WP_Data object (WC_Coupon for coupons, etc). + * @param string $meta_key Meta key to update. + * @param mixed $meta_value Value to save. + * + * @since 3.6.0 Added to prevent empty meta being stored unless required. + * + * @return bool True if updated/deleted. + */ + protected function update_or_delete_meta( $object, $meta_key, $meta_value ) { + if ( in_array( $meta_value, array( array(), '' ), true ) && ! in_array( $meta_key, $this->must_exist_meta_keys, true ) ) { + $updated = delete_metadata( $this->meta_type, $object->get_id(), $meta_key ); + } else { + $updated = update_metadata( $this->meta_type, $object->get_id(), $meta_key, $meta_value ); + } + + return (bool) $updated; + } + + /** + * Get valid WP_Query args from a WC_Order_Query's query variables. + * + * @since 3.0.6 + * @param array $query_vars query vars from a WC_Order_Query. + * @return array + */ + protected function get_wp_query_args( $query_vars ) { + global $wpdb; + + $wp_query_args = parent::get_wp_query_args( $query_vars ); + + // Force type to be existent + if ( isset( $query_vars['type'] ) ) { + $wp_query_args['type'] = $query_vars['type']; + } + + if ( ! isset( $wp_query_args['date_query'] ) ) { + $wp_query_args['date_query'] = array(); + } + + if ( ! isset( $wp_query_args['meta_query'] ) ) { + $wp_query_args['meta_query'] = array(); // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query + } + + // Allow Woo to treat these props as date query compatible + $date_queries = array( + 'date_created' => 'post_date', + ); + + foreach ( $date_queries as $query_var_key => $db_key ) { + if ( isset( $query_vars[ $query_var_key ] ) && '' !== $query_vars[ $query_var_key ] ) { + + // Remove any existing meta queries for the same keys to prevent conflicts. + $existing_queries = wp_list_pluck( $wp_query_args['meta_query'], 'key', true ); + $meta_query_index = array_search( $db_key, $existing_queries, true ); + + if ( false !== $meta_query_index ) { + unset( $wp_query_args['meta_query'][ $meta_query_index ] ); + } + + $wp_query_args = $this->parse_date_for_wp_query( $query_vars[ $query_var_key ], $db_key, $wp_query_args ); + } + } + + /** + * Replace date query columns after Woo parsed dates. + * Include table name because otherwise WP_Date_Query won't accept our custom column. + */ + if ( isset( $wp_query_args['date_query'] ) ) { + foreach ( $wp_query_args['date_query'] as $key => $date_query ) { + if ( isset( $date_query['column'] ) && in_array( $date_query['column'], $date_queries, true ) ) { + $wp_query_args['date_query'][ $key ]['column'] = $wpdb->gzd_shipment_labels . '.label_' . array_search( $date_query['column'], $date_queries, true ); + } + } + } + + if ( ! isset( $query_vars['paginate'] ) || ! $query_vars['paginate'] ) { + $wp_query_args['no_found_rows'] = true; + } + + /** + * Filter to adjust the DHL label query args after parsing them. + * + * @param array $wp_query_args Parsed query arguments. + * @param array $query_vars Original query arguments. + * @param Label $data_store The label data store. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + return apply_filters( 'woocommerce_gzd_shipment_label_data_store_get_labels_query', $wp_query_args, $query_vars, $this ); + } + + public function get_query_args( $query_vars ) { + return $this->get_wp_query_args( $query_vars ); + } + + /** + * Table structure is slightly different between meta types, this function will return what we need to know. + * + * @since 3.0.0 + * @return array Array elements: table, object_id_field, meta_id_field + */ + protected function get_db_info() { + global $wpdb; + + $meta_id_field = 'meta_id'; // for some reason users calls this umeta_id so we need to track this as well. + $table = $wpdb->gzd_shipment_labelmeta; + $object_id_field = $this->meta_type . '_id'; + + if ( ! empty( $this->object_id_field_for_meta ) ) { + $object_id_field = $this->object_id_field_for_meta; + } + + return array( + 'table' => $table, + 'object_id_field' => $object_id_field, + 'meta_id_field' => $meta_id_field, + ); + } + + public function get_label_count() { + global $wpdb; + + return absint( $wpdb->get_var( "SELECT COUNT( * ) FROM {$wpdb->gzd_shipment_labels}" ) ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/DataStores/Packaging.php b/packages/woocommerce-germanized-shipments/src/DataStores/Packaging.php new file mode 100644 index 000000000..10658d0f3 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/DataStores/Packaging.php @@ -0,0 +1,696 @@ +set_date_created( time() ); + + $data = array( + 'packaging_type' => $packaging->get_type(), + 'packaging_description' => $packaging->get_description(), + 'packaging_weight' => $packaging->get_weight(), + 'packaging_max_content_weight' => $packaging->get_max_content_weight(), + 'packaging_length' => $packaging->get_length(), + 'packaging_width' => $packaging->get_width(), + 'packaging_height' => $packaging->get_height(), + 'packaging_order' => $packaging->get_order(), + 'packaging_date_created' => gmdate( 'Y-m-d H:i:s', $packaging->get_date_created( 'edit' )->getOffsetTimestamp() ), + 'packaging_date_created_gmt' => gmdate( 'Y-m-d H:i:s', $packaging->get_date_created( 'edit' )->getTimestamp() ), + ); + + $wpdb->insert( + $wpdb->gzd_packaging, + $data + ); + + $packaging_id = $wpdb->insert_id; + + if ( $packaging_id ) { + $packaging->set_id( $packaging_id ); + + $this->save_packaging_data( $packaging ); + + $packaging->save_meta_data(); + $packaging->apply_changes(); + + $this->clear_caches( $packaging ); + + /** + * Action that indicates that a new Packaging has been created in the DB. + * + * @param integer $packaging_id The packaging id. + * @param \Vendidero\Germanized\Shipments\Packaging $packaging The packaging instance. + * + * @since 3.3.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_new_packaging', $packaging_id, $packaging ); + } + } + + /** + * Method to update a packaging in the database. + * + * @param \Vendidero\Germanized\Shipments\Packaging $packaging Packaging object. + */ + public function update( &$packaging ) { + global $wpdb; + + $updated_props = array(); + $core_props = $this->core_props; + $changed_props = array_keys( $packaging->get_changes() ); + $packaging_data = array(); + + foreach ( $changed_props as $prop ) { + + if ( ! in_array( $prop, $core_props, true ) ) { + continue; + } + + switch ( $prop ) { + case 'date_created': + if ( is_callable( array( $packaging, 'get_' . $prop ) ) ) { + $packaging_data[ 'packaging_' . $prop ] = gmdate( 'Y-m-d H:i:s', $packaging->{'get_' . $prop}( 'edit' )->getOffsetTimestamp() ); + $packaging_data[ 'packaging_' . $prop . '_gmt' ] = gmdate( 'Y-m-d H:i:s', $packaging->{'get_' . $prop}( 'edit' )->getTimestamp() ); + } + break; + default: + if ( is_callable( array( $packaging, 'get_' . $prop ) ) ) { + $packaging_data[ 'packaging_' . $prop ] = $packaging->{'get_' . $prop}( 'edit' ); + } + break; + } + } + + if ( ! empty( $packaging_data ) ) { + $wpdb->update( + $wpdb->gzd_packaging, + $packaging_data, + array( 'packaging_id' => $packaging->get_id() ) + ); + } + + $this->save_packaging_data( $packaging ); + + $packaging->save_meta_data(); + $packaging->apply_changes(); + + $this->clear_caches( $packaging ); + + /** + * Action that indicates that a Packaging has been updated in the DB. + * + * @param integer $packaging_id The packaging id. + * @param \Vendidero\Germanized\Shipments\Packaging $packaging The packaging instance. + * + * @since 3.3.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_packaging_updated', $packaging->get_id(), $packaging ); + } + + /** + * Remove a Packaging from the database. + * + * @since 3.0.0 + * @param \Vendidero\Germanized\Shipments\Packaging $packaging Packaging object. + * @param bool $force_delete Unused param. + */ + public function delete( &$packaging, $force_delete = false ) { + global $wpdb; + + $wpdb->delete( $wpdb->gzd_packaging, array( 'packaging_id' => $packaging->get_id() ), array( '%d' ) ); + $wpdb->delete( $wpdb->gzd_packagingmeta, array( 'gzd_packaging_id' => $packaging->get_id() ), array( '%d' ) ); + + $this->clear_caches( $packaging ); + + /** + * Action that indicates that a Packaging has been deleted from the DB. + * + * @param integer $packaging_id The packaging id. + * @param \Vendidero\Germanized\Shipments\Packaging $packaging The packaging instance. + * + * @since 3.3.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_packaging_deleted', $packaging->get_id(), $packaging ); + } + + /** + * Read a Packaging from the database. + * + * @since 3.3.0 + * + * @param \Vendidero\Germanized\Shipments\Packaging $packaging Packaging object. + * + * @throws Exception Throw exception if invalid packaging. + */ + public function read( &$packaging ) { + global $wpdb; + + $data = $wpdb->get_row( + $wpdb->prepare( + "SELECT * FROM {$wpdb->gzd_packaging} WHERE packaging_id = %d LIMIT 1", + $packaging->get_id() + ) + ); + + if ( $data ) { + $packaging->set_props( + array( + 'type' => $data->packaging_type, + 'description' => $data->packaging_description, + 'weight' => $data->packaging_weight, + 'max_content_weight' => $data->packaging_max_content_weight, + 'length' => $data->packaging_length, + 'width' => $data->packaging_width, + 'height' => $data->packaging_height, + 'order' => $data->packaging_order, + 'date_created' => '0000-00-00 00:00:00' !== $data->packaging_date_created_gmt ? wc_string_to_timestamp( $data->packaging_date_created_gmt ) : null, + ) + ); + + $this->read_packaging_data( $packaging ); + + $packaging->read_meta_data(); + $packaging->set_object_read( true ); + + /** + * Action that indicates that a Packaging has been loaded from DB. + * + * @param \Vendidero\Germanized\Shipments\Packaging $packaging The Packaging object. + * + * @since 3.3.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_packaging_loaded', $packaging ); + } else { + throw new Exception( _x( 'Invalid packaging.', 'shipments', 'woocommerce-germanized' ) ); + } + } + + /** + * Clear any caches. + * + * @param \Vendidero\Germanized\Shipments\Packaging $packaging Packaging object. + * @since 3.0.0 + */ + protected function clear_caches( &$packaging ) { + wp_cache_delete( $packaging->get_id(), $this->meta_type . '_meta' ); + wp_cache_delete( 'packaging-list', 'packaging' ); + } + + /* + |-------------------------------------------------------------------------- + | Additional Methods + |-------------------------------------------------------------------------- + */ + + /** + * Get the packaging type based on ID. + * + * @param int $packaging_id Packaging id. + * @return string + */ + public function get_packaging_type( $packing_id ) { + global $wpdb; + + $type = $wpdb->get_col( + $wpdb->prepare( + "SELECT packaging_type FROM {$wpdb->gzd_packaging} WHERE packaging_id = %d LIMIT 1", + $packing_id + ) + ); + + return ! empty( $type ) ? $type[0] : false; + } + + /** + * Read extra data associated with the Packaging. + * + * @param \Vendidero\Germanized\Shipments\Packaging $packaging Packaging object. + * @since 3.0.0 + */ + protected function read_packaging_data( &$packaging ) { + $props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + $props[ substr( $meta_key, 1 ) ] = get_metadata( 'gzd_packaging', $packaging->get_id(), $meta_key, true ); + } + + $packaging->set_props( $props ); + } + + /** + * @param \Vendidero\Germanized\Shipments\Packaging $packaging + */ + protected function save_packaging_data( &$packaging ) { + $updated_props = array(); + $meta_key_to_props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + $prop_name = substr( $meta_key, 1 ); + + if ( in_array( $prop_name, $this->core_props, true ) ) { + continue; + } + + $meta_key_to_props[ $meta_key ] = $prop_name; + } + + $props_to_update = $this->get_props_to_update( $packaging, $meta_key_to_props, $this->meta_type ); + + foreach ( $props_to_update as $meta_key => $prop ) { + + if ( ! is_callable( array( $packaging, "get_$prop" ) ) ) { + continue; + } + + $value = $packaging->{"get_$prop"}( 'edit' ); + $value = is_string( $value ) ? wp_slash( $value ) : $value; + + $updated = $this->update_or_delete_meta( $packaging, $meta_key, $value ); + + if ( $updated ) { + $updated_props[] = $prop; + } + } + + /** + * Action that fires after updating a Packaging's properties. + * + * @param \Vendidero\Germanized\Shipments\Packaging $packaging The Packaging object. + * @param array $changed_props The updated properties. + * + * @since 3.3.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_packaging_object_updated_props', $packaging, $updated_props ); + } + + /** + * Update meta data in, or delete it from, the database. + * + * Avoids storing meta when it's either an empty string or empty array. + * Other empty values such as numeric 0 and null should still be stored. + * Data-stores can force meta to exist using `must_exist_meta_keys`. + * + * Note: WordPress `get_metadata` function returns an empty string when meta data does not exist. + * + * @param WC_Data $object The WP_Data object (WC_Coupon for coupons, etc). + * @param string $meta_key Meta key to update. + * @param mixed $meta_value Value to save. + * + * @since 3.6.0 Added to prevent empty meta being stored unless required. + * + * @return bool True if updated/deleted. + */ + protected function update_or_delete_meta( $object, $meta_key, $meta_value ) { + if ( in_array( $meta_value, array( array(), '' ), true ) && ! in_array( $meta_key, $this->must_exist_meta_keys, true ) ) { + $updated = delete_metadata( $this->meta_type, $object->get_id(), $meta_key ); + } else { + $updated = update_metadata( $this->meta_type, $object->get_id(), $meta_key, $meta_value ); + } + + return (bool) $updated; + } + + /** + * Get valid WP_Query args from a WC_Order_Query's query variables. + * + * @since 3.0.6 + * @param array $query_vars query vars from a WC_Order_Query. + * @return array + */ + protected function get_wp_query_args( $query_vars ) { + global $wpdb; + + $wp_query_args = parent::get_wp_query_args( $query_vars ); + + // Force type to be existent + if ( isset( $query_vars['type'] ) ) { + $wp_query_args['type'] = $query_vars['type']; + } + + if ( ! isset( $wp_query_args['date_query'] ) ) { + $wp_query_args['date_query'] = array(); + } + + if ( ! isset( $wp_query_args['meta_query'] ) ) { + $wp_query_args['meta_query'] = array(); // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query + } + + // Allow Woo to treat these props as date query compatible + $date_queries = array( + 'date_created', + ); + + foreach ( $date_queries as $db_key ) { + if ( isset( $query_vars[ $db_key ] ) && '' !== $query_vars[ $db_key ] ) { + + // Remove any existing meta queries for the same keys to prevent conflicts. + $existing_queries = wp_list_pluck( $wp_query_args['meta_query'], 'key', true ); + $meta_query_index = array_search( $db_key, $existing_queries, true ); + + if ( false !== $meta_query_index ) { + unset( $wp_query_args['meta_query'][ $meta_query_index ] ); + } + + $date_query_args = $this->parse_date_for_wp_query( $query_vars[ $db_key ], 'post_date', array() ); + + /** + * Replace date query columns after Woo parsed dates. + * Include table name because otherwise WP_Date_Query won't accept our custom column. + */ + if ( isset( $date_query_args['date_query'] ) && ! empty( $date_query_args['date_query'] ) ) { + $date_query = $date_query_args['date_query'][0]; + + if ( 'post_date' === $date_query['column'] ) { + $date_query['column'] = $wpdb->gzd_shipments . '.shipment_' . $db_key; + } + + $wp_query_args['date_query'][] = $date_query; + } + } + } + + if ( ! isset( $query_vars['paginate'] ) || ! $query_vars['paginate'] ) { + $wp_query_args['no_found_rows'] = true; + } + + /** + * Filter to adjust Packaging query arguments after parsing. + * + * @param array $wp_query_args Array containing parsed query arguments. + * @param array $query_vars The original query arguments. + * @param Packaging $data_store The packaging data store object. + * + * @since 3.3.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_packaging_data_store_get_shipments_query', $wp_query_args, $query_vars, $this ); + } + + public function get_packaging_list( $args = array() ) { + global $wpdb; + + $all_types = array_keys( wc_gzd_get_packaging_types() ); + + $args = wp_parse_args( + $args, + array( + 'type' => $all_types, + ) + ); + + if ( ! is_array( $args['type'] ) ) { + $args['type'] = array( $args['type'] ); + } + + $types = array_filter( wc_clean( $args['type'] ) ); + $types = empty( $types ) ? $all_types : $types; + + $query = " + SELECT packaging_id FROM {$wpdb->gzd_packaging} + WHERE packaging_type IN ( '" . implode( "','", $types ) . "' ) + ORDER BY packaging_order ASC + "; + + if ( $all_types === $types ) { + // Get from cache if available. + $results = wp_cache_get( 'packaging-list', 'packaging' ); + + if ( false === $results ) { + $results = $wpdb->get_results( $query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + + wp_cache_set( 'packaging-list', $results, 'packaging' ); + } + } else { + $results = $wpdb->get_results( $query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } + + foreach ( $results as $key => $packaging ) { + $results[ $key ] = wc_gzd_get_packaging( $packaging ); + } + + return $results; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * @param \Vendidero\Germanized\Shipments\Packaging|PackagingBox $packaging + * + * @return bool + */ + public function shipment_fits_into_packaging_naive( $shipment, $packaging ) { + if ( is_a( $packaging, '\Vendidero\Germanized\Shipments\Packing\PackagingBox' ) ) { + $packaging = $packaging->getReference(); + } + + if ( ! $packaging ) { + return false; + } + + $weight = (float) wc_format_decimal( empty( $shipment->get_content_weight() ) ? 0 : wc_get_weight( $shipment->get_content_weight(), wc_gzd_get_packaging_weight_unit(), $shipment->get_weight_unit() ), 1 ); + $volume = (float) wc_format_decimal( empty( $shipment->get_content_volume() ) ? 0 : wc_gzd_get_volume_dimension( $shipment->get_content_volume(), wc_gzd_get_packaging_dimension_unit(), $shipment->get_dimension_unit() ), 1 ); + $fits = true; + $packaging_volume = (float) $packaging->get_length() * (float) $packaging->get_width() * (float) $packaging->get_height(); + + /** + * The packaging does not fit in case: + * - total weight is greater than it's maximum capability + * - the total volume is greater than the packaging volume + */ + if ( ! empty( $packaging->get_max_content_weight() ) && $weight > $packaging->get_max_content_weight() ) { + $fits = false; + } elseif ( $volume > $packaging_volume ) { + $fits = false; + } + + return $fits; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * + * @return \Vendidero\Germanized\Shipments\Packaging[] + */ + public function find_available_packaging_for_shipment( $shipment ) { + $packaging_available = array(); + $items_to_pack = $shipment->get_items_to_pack(); + $results = false; + + // Get from cache if available. + if ( $shipment->get_id() > 0 ) { + $results = wp_cache_get( 'available-packaging-' . $shipment->get_id(), 'shipments' ); + } + + if ( false === $results && count( $items_to_pack ) > 0 ) { + $available_packaging_ids = array(); + + if ( Package::is_packing_supported() ) { + $packaging_list = wc_gzd_get_packaging_list(); + $items = \DVDoug\BoxPacker\ItemList::fromArray( $items_to_pack ); + + foreach ( $packaging_list as $packaging ) { + /** + * Make sure to only check naively fitting packaging to improve performance + */ + if ( ! $this->shipment_fits_into_packaging_naive( $shipment, $packaging ) ) { + continue; + } + + $box = new PackagingBox( $packaging ); + $org_size = count( $items_to_pack ); + + $packer = new VolumePacker( $box, $items ); + $packed = $packer->pack(); + + if ( count( $packed->getItems() ) === $org_size ) { + $packaging_available[] = $packaging; + $available_packaging_ids[] = $packaging->get_id(); + } + } + } else { + global $wpdb; + + $weight = wc_format_decimal( empty( $shipment->get_weight() ) ? 0 : wc_get_weight( $shipment->get_weight(), wc_gzd_get_packaging_weight_unit(), $shipment->get_weight_unit() ), 1 ); + $length = wc_format_decimal( empty( $shipment->get_length() ) ? 0 : wc_get_dimension( $shipment->get_length(), wc_gzd_get_packaging_dimension_unit(), $shipment->get_dimension_unit() ), 1 ); + $width = wc_format_decimal( empty( $shipment->get_width() ) ? 0 : wc_get_dimension( $shipment->get_width(), wc_gzd_get_packaging_dimension_unit(), $shipment->get_dimension_unit() ), 1 ); + $height = wc_format_decimal( empty( $shipment->get_height() ) ? 0 : wc_get_dimension( $shipment->get_height(), wc_gzd_get_packaging_dimension_unit(), $shipment->get_dimension_unit() ), 1 ); + + $types = array_keys( wc_gzd_get_packaging_types() ); + $threshold = apply_filters( 'woocommerce_gzd_shipment_packaging_match_threshold', 0, $shipment ); + + $query_sql = "SELECT + packaging_id, + (packaging_length - %f) AS length_diff, + (packaging_width - %f) AS width_diff, + (packaging_height - %f) AS height_diff, + ((packaging_length - %f) + (packaging_width - %f) + (packaging_height - %f)) AS total_diff + FROM {$wpdb->gzd_packaging} + WHERE ( packaging_max_content_weight = 0 OR packaging_max_content_weight >= %f ) AND packaging_type IN ( '" . implode( "','", $types ) . "' ) + HAVING length_diff >= %f AND width_diff >= %f AND height_diff >= %f + ORDER BY total_diff ASC, packaging_weight ASC, packaging_order ASC + "; + + $query = $wpdb->prepare( $query_sql, $length, $width, $height, $length, $width, $height, $weight, $threshold, $threshold, $threshold ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + $results = $wpdb->get_results( $query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + + if ( $results ) { + foreach ( $results as $result ) { + $available_packaging_ids[] = $result->packaging_id; + $packaging_available[] = wc_gzd_get_packaging( $result->packaging_id ); + } + } + } + + wp_cache_set( 'available-packaging-' . $shipment->get_id(), $available_packaging_ids, 'shipments' ); + } elseif ( count( $items_to_pack ) <= 0 ) { + $packaging_available = wc_gzd_get_packaging_list(); + } else { + foreach ( (array) $results as $packaging_id ) { + $packaging_available[] = wc_gzd_get_packaging( $packaging_id ); + } + } + + $packaging_list = apply_filters( 'woocommerce_gzd_find_available_packaging_for_shipment', $packaging_available, $shipment ); + + return $this->sort_packaging_list( $packaging_list ); + } + + protected function sort_packaging_list( $packaging ) { + usort( $packaging, array( $this, 'sort_packaging_list_callback' ) ); + + return $packaging; + } + + /** + * @param \Vendidero\Germanized\Shipments\Packaging $packaging_a + * @param \Vendidero\Germanized\Shipments\Packaging $packaging_b + * + * @return int + */ + protected function sort_packaging_list_callback( $packaging_a, $packaging_b ) { + if ( $packaging_a->get_volume() === $packaging_b->get_volume() ) { + return 0; + } + + return ( $packaging_a->get_volume() > $packaging_b->get_volume() ) ? 1 : -1; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * @param string|array $types + * + * @return \Vendidero\Germanized\Shipments\Packaging|false + */ + public function find_best_match_for_shipment( $shipment ) { + $results = $this->find_available_packaging_for_shipment( $shipment ); + $packaging = false; + + if ( ! empty( $results ) ) { + $packaging = $results[0]; + + /** + * In case more than one packaging is available - choose the default packaging in case it is available. + */ + if ( count( $results ) > 1 ) { + $default_packaging_id = Package::get_setting( 'default_packaging' ); + + if ( ! empty( $default_packaging_id ) ) { + $default_packaging_id = absint( $default_packaging_id ); + + foreach ( $results as $result ) { + if ( $result->get_id() === $default_packaging_id ) { + $packaging = $result; + break; + } + } + } + } + } + + return apply_filters( 'woocommerce_gzd_find_best_matching_packaging_for_shipment', $packaging, $shipment ); + } + + /** + * Table structure is slightly different between meta types, this function will return what we need to know. + * + * @since 3.0.0 + * @return array Array elements: table, object_id_field, meta_id_field + */ + protected function get_db_info() { + global $wpdb; + + $meta_id_field = 'meta_id'; // for some reason users calls this umeta_id so we need to track this as well. + $table = $wpdb->gzd_packagingmeta; + $object_id_field = $this->meta_type . '_id'; + + if ( ! empty( $this->object_id_field_for_meta ) ) { + $object_id_field = $this->object_id_field_for_meta; + } + + return array( + 'table' => $table, + 'object_id_field' => $object_id_field, + 'meta_id_field' => $meta_id_field, + ); + } + + public function get_query_args( $query_vars ) { + return $this->get_wp_query_args( $query_vars ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/DataStores/Shipment.php b/packages/woocommerce-germanized-shipments/src/DataStores/Shipment.php new file mode 100644 index 000000000..3011ca443 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/DataStores/Shipment.php @@ -0,0 +1,720 @@ +set_date_created( time() ); + $shipment->set_weight_unit( get_option( 'woocommerce_weight_unit', 'kg' ) ); + $shipment->set_dimension_unit( get_option( 'woocommerce_dimension_unit', 'cm' ) ); + + $data = array( + 'shipment_country' => $shipment->get_country(), + 'shipment_order_id' => is_callable( array( $shipment, 'get_order_id' ) ) ? $shipment->get_order_id() : 0, + 'shipment_parent_id' => is_callable( array( $shipment, 'get_parent_id' ) ) ? $shipment->get_parent_id() : 0, + 'shipment_tracking_id' => $shipment->get_tracking_id(), + 'shipment_status' => $this->get_status( $shipment ), + 'shipment_search_index' => $this->get_search_index( $shipment ), + 'shipment_packaging_id' => $shipment->get_packaging_id(), + 'shipment_type' => $shipment->get_type(), + 'shipment_shipping_provider' => $shipment->get_shipping_provider(), + 'shipment_shipping_method' => $shipment->get_shipping_method(), + 'shipment_date_created' => gmdate( 'Y-m-d H:i:s', $shipment->get_date_created( 'edit' )->getOffsetTimestamp() ), + 'shipment_date_created_gmt' => gmdate( 'Y-m-d H:i:s', $shipment->get_date_created( 'edit' )->getTimestamp() ), + 'shipment_version' => Package::get_version(), + ); + + if ( $shipment->get_date_sent() ) { + $data['shipment_date_sent'] = gmdate( 'Y-m-d H:i:s', $shipment->get_date_sent( 'edit' )->getOffsetTimestamp() ); + $data['shipment_date_sent_gmt'] = gmdate( 'Y-m-d H:i:s', $shipment->get_date_sent( 'edit' )->getTimestamp() ); + } + + if ( is_callable( array( $shipment, 'get_est_delivery_date' ) ) && $shipment->get_est_delivery_date() ) { + $data['shipment_est_delivery_date'] = gmdate( 'Y-m-d H:i:s', $shipment->get_est_delivery_date( 'edit' )->getOffsetTimestamp() ); + $data['shipment_est_delivery_date_gmt'] = gmdate( 'Y-m-d H:i:s', $shipment->get_est_delivery_date( 'edit' )->getTimestamp() ); + } + + $wpdb->insert( + $wpdb->gzd_shipments, + $data + ); + + $shipment_id = $wpdb->insert_id; + + if ( $shipment_id ) { + $shipment->set_id( $shipment_id ); + + $this->save_shipment_data( $shipment ); + + $shipment->save_meta_data(); + $shipment->apply_changes(); + + $this->clear_caches( $shipment ); + + $hook_postfix = $this->get_hook_postfix( $shipment ); + + /** + * Action that indicates that a new Shipment has been created in the DB. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * shipment type in case it is not a simple shipment. + * + * @param integer $shipment_id The shipment id. + * @param Shipment $shipment The shipment instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "woocommerce_gzd_new_{$hook_postfix}shipment", $shipment_id, $shipment ); + } + } + + /** + * Get the status to save to the object. + * + * @since 3.6.0 + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + * @return string + */ + protected function get_status( $shipment ) { + $shipment_status = $shipment->get_status( 'edit' ); + + if ( ! $shipment_status ) { + /** This filter is documented in src/Shipment.php */ + $shipment_status = apply_filters( 'woocommerce_gzd_get_shipment_default_status', 'gzd-draft' ); + } + + $valid_statuses = array_keys( wc_gzd_get_shipment_statuses() ); + + // Add a gzd- prefix to the status. + if ( in_array( 'gzd-' . $shipment_status, $valid_statuses, true ) ) { + $shipment_status = 'gzd-' . $shipment_status; + } + + return $shipment_status; + } + + /** + * Method to update a shipment in the database. + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + */ + public function update( &$shipment ) { + global $wpdb; + + $updated_props = array(); + $core_props = $this->core_props; + $changed_props = array_keys( $shipment->get_changes() ); + $shipment_data = array(); + + if ( '' === $shipment->get_weight_unit( 'edit' ) ) { + $shipment->set_weight_unit( get_option( 'woocommerce_weight_unit', 'kg' ) ); + } + + if ( '' === $shipment->get_dimension_unit( 'edit' ) ) { + $shipment->set_dimension_unit( get_option( 'woocommerce_dimension_unit', 'cm' ) ); + } + + // Make sure country in core props is updated as soon as the address changes + if ( in_array( 'address', $changed_props, true ) ) { + $changed_props[] = 'country'; + + // Update search index + $shipment_data['shipment_search_index'] = $this->get_search_index( $shipment ); + } + + // Shipping provider has changed - lets remove existing label + if ( in_array( 'shipping_provider', $changed_props, true ) ) { + + if ( $shipment->supports_label() && $shipment->has_label() ) { + $shipment->get_label()->delete(); + } + } + + foreach ( $changed_props as $prop ) { + + if ( ! in_array( $prop, $core_props, true ) ) { + continue; + } + + switch ( $prop ) { + case 'status': + $shipment_data[ 'shipment_' . $prop ] = $this->get_status( $shipment ); + break; + case 'date_created': + case 'date_sent': + case 'est_delivery_date': + if ( is_callable( array( $shipment, 'get_' . $prop ) ) ) { + $shipment_data[ 'shipment_' . $prop ] = gmdate( 'Y-m-d H:i:s', $shipment->{'get_' . $prop}( 'edit' )->getOffsetTimestamp() ); + $shipment_data[ 'shipment_' . $prop . '_gmt' ] = gmdate( 'Y-m-d H:i:s', $shipment->{'get_' . $prop}( 'edit' )->getTimestamp() ); + } + break; + default: + if ( is_callable( array( $shipment, 'get_' . $prop ) ) ) { + $shipment_data[ 'shipment_' . $prop ] = $shipment->{'get_' . $prop}( 'edit' ); + } + break; + } + } + + if ( ! empty( $shipment_data ) ) { + $shipment_data['shipment_search_index'] = $this->get_search_index( $shipment ); + + $wpdb->update( + $wpdb->gzd_shipments, + $shipment_data, + array( 'shipment_id' => $shipment->get_id() ) + ); + } + + $this->save_shipment_data( $shipment ); + + $shipment->save_meta_data(); + $shipment->apply_changes(); + + $this->clear_caches( $shipment ); + + $hook_postfix = $this->get_hook_postfix( $shipment ); + + /** + * Action that indicates that a Shipment has been updated in the DB. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * shipment type in case it is not a simple shipment. + * + * @param integer $shipment_id The shipment id. + * @param Shipment $shipment The shipment instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "woocommerce_gzd_{$hook_postfix}shipment_updated", $shipment->get_id(), $shipment ); + } + + /** + * Remove a shipment from the database. + * + * @since 3.0.0 + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + * @param bool $force_delete Unused param. + */ + public function delete( &$shipment, $force_delete = false ) { + global $wpdb; + + $wpdb->delete( $wpdb->gzd_shipments, array( 'shipment_id' => $shipment->get_id() ), array( '%d' ) ); + $wpdb->delete( $wpdb->gzd_shipmentmeta, array( 'gzd_shipment_id' => $shipment->get_id() ), array( '%d' ) ); + + $this->delete_items( $shipment ); + $this->clear_caches( $shipment ); + + $hook_postfix = $this->get_hook_postfix( $shipment ); + + /** + * Action that indicates that a Shipment has been deleted from the DB. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * shipment type in case it is not a simple shipment. + * + * @param integer $shipment_id The shipment id. + * @param \Vendidero\Germanized\Shipments\Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "woocommerce_gzd_{$hook_postfix}shipment_deleted", $shipment->get_id(), $shipment ); + } + + /** + * Read a shipment from the database. + * + * @since 3.0.0 + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + * + * @throws Exception Throw exception if invalid shipment. + */ + public function read( &$shipment ) { + global $wpdb; + + $data = $wpdb->get_row( + $wpdb->prepare( + "SELECT * FROM {$wpdb->gzd_shipments} WHERE shipment_id = %d LIMIT 1", + $shipment->get_id() + ) + ); + + if ( $data ) { + $shipment->set_props( + array( + 'order_id' => $data->shipment_order_id, + 'parent_id' => $data->shipment_parent_id, + 'country' => $data->shipment_country, + 'tracking_id' => $data->shipment_tracking_id, + 'shipping_provider' => $data->shipment_shipping_provider, + 'shipping_method' => $data->shipment_shipping_method, + 'packaging_id' => $data->shipment_packaging_id, + 'date_created' => $this->is_valid_timestamp( $data->shipment_date_created_gmt ) ? wc_string_to_timestamp( $data->shipment_date_created_gmt ) : null, + 'date_sent' => $this->is_valid_timestamp( $data->shipment_date_sent_gmt ) ? wc_string_to_timestamp( $data->shipment_date_sent_gmt ) : null, + 'est_delivery_date' => $this->is_valid_timestamp( $data->shipment_est_delivery_date_gmt ) ? wc_string_to_timestamp( $data->shipment_est_delivery_date_gmt ) : null, + 'status' => $data->shipment_status, + 'version' => $data->shipment_version, + ) + ); + + $this->read_shipment_data( $shipment ); + + $shipment->read_meta_data(); + $shipment->set_object_read( true ); + + $hook_postfix = $this->get_hook_postfix( $shipment ); + + /** + * Action that indicates that a Shipment has been loaded from DB. + * + * The dynamic portion of this hook, `$hook_postfix` refers to the + * shipment type in case it is not a simple shipment. + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "woocommerce_gzd_{$hook_postfix}shipment_loaded", $shipment ); + } else { + throw new Exception( _x( 'Invalid shipment.', 'shipments', 'woocommerce-germanized' ) ); + } + } + + protected function is_valid_timestamp( $mysql_date ) { + return ( '0000-00-00 00:00:00' === $mysql_date || null === $mysql_date ) ? false : true; + } + + /** + * Clear any caches. + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + * @since 3.0.0 + */ + protected function clear_caches( &$shipment ) { + wp_cache_delete( 'shipment-items-' . $shipment->get_id(), 'shipments' ); + wp_cache_delete( $shipment->get_id(), $this->meta_type . '_meta' ); + wp_cache_delete( 'available-packaging-' . $shipment->get_id(), 'shipments' ); + } + + /* + |-------------------------------------------------------------------------- + | Additional Methods + |-------------------------------------------------------------------------- + */ + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + protected function get_search_index( $shipment ) { + $index = array(); + + if ( is_a( $shipment, '\Vendidero\Germanized\Shipments\ReturnShipment' ) ) { + $index = array_merge( $index, $shipment->get_sender_address() ); + } else { + $index = array_merge( $index, $shipment->get_address() ); + } + + return implode( ' ', $index ); + } + + protected function get_hook_postfix( $shipment ) { + if ( 'simple' !== $shipment->get_type() ) { + return $shipment->get_type() . '_'; + } + + return ''; + } + + /** + * Get the label type based on label ID. + * + * @param int $shipment_id Shipment id. + * @return string + */ + public function get_shipment_type( $shipment_id ) { + global $wpdb; + + $type = $wpdb->get_col( + $wpdb->prepare( + "SELECT shipment_type FROM {$wpdb->gzd_shipments} WHERE shipment_id = %d LIMIT 1", + $shipment_id + ) + ); + + return ! empty( $type ) ? $type[0] : false; + } + + /** + * Read extra data associated with the shipment. + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + * @since 3.0.0 + */ + protected function read_shipment_data( &$shipment ) { + $props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + $props[ substr( $meta_key, 1 ) ] = get_metadata( 'gzd_shipment', $shipment->get_id(), $meta_key, true ); + } + + $shipment->set_props( $props ); + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + protected function save_shipment_data( &$shipment ) { + $updated_props = array(); + $meta_key_to_props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + $prop_name = substr( $meta_key, 1 ); + + if ( in_array( $prop_name, $this->core_props, true ) ) { + continue; + } + + $meta_key_to_props[ $meta_key ] = $prop_name; + } + + $props_to_update = $this->get_props_to_update( $shipment, $meta_key_to_props, 'gzd_shipment' ); + + foreach ( $props_to_update as $meta_key => $prop ) { + + if ( ! is_callable( array( $shipment, "get_$prop" ) ) ) { + continue; + } + + $value = $shipment->{"get_$prop"}( 'edit' ); + $value = is_string( $value ) ? wp_slash( $value ) : $value; + + switch ( $prop ) { + case 'is_customer_requested': + $value = wc_bool_to_string( $value ); + break; + } + + // Force updating props that are dependent on inner content data (weight, dimensions) + if ( in_array( $prop, array( 'weight', 'width', 'length', 'height' ), true ) && ! $shipment->is_editable() ) { + + // Get weight in view context to maybe allow calculating inner content props. + $value = $shipment->{"get_$prop"}( 'view' ); + $updated = update_metadata( 'gzd_shipment', $shipment->get_id(), $meta_key, $value ); + } else { + $updated = $this->update_or_delete_meta( $shipment, $meta_key, $value ); + } + + if ( $updated ) { + $updated_props[] = $prop; + } + } + + /** + * Action that fires after updating a Shipment's properties. + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment The shipment object. + * @param array $changed_props The updated properties. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_object_updated_props', $shipment, $updated_props ); + } + + /** + * Update meta data in, or delete it from, the database. + * + * Avoids storing meta when it's either an empty string or empty array. + * Other empty values such as numeric 0 and null should still be stored. + * Data-stores can force meta to exist using `must_exist_meta_keys`. + * + * Note: WordPress `get_metadata` function returns an empty string when meta data does not exist. + * + * @param WC_Data $object The WP_Data object (WC_Coupon for coupons, etc). + * @param string $meta_key Meta key to update. + * @param mixed $meta_value Value to save. + * + * @since 3.6.0 Added to prevent empty meta being stored unless required. + * + * @return bool True if updated/deleted. + */ + protected function update_or_delete_meta( $object, $meta_key, $meta_value ) { + if ( in_array( $meta_value, array( array(), '' ), true ) && ! in_array( $meta_key, $this->must_exist_meta_keys, true ) ) { + $updated = delete_metadata( 'gzd_shipment', $object->get_id(), $meta_key ); + } else { + $updated = update_metadata( 'gzd_shipment', $object->get_id(), $meta_key, $meta_value ); + } + + return (bool) $updated; + } + + /** + * Read items from the database for this shipment. + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + * + * @return array + */ + public function read_items( $shipment ) { + global $wpdb; + + // Get from cache if available. + $items = 0 < $shipment->get_id() ? wp_cache_get( 'shipment-items-' . $shipment->get_id(), 'shipments' ) : false; + + if ( false === $items ) { + + $items = $wpdb->get_results( + $wpdb->prepare( "SELECT * FROM {$wpdb->gzd_shipment_items} WHERE shipment_id = %d ORDER BY shipment_item_id;", $shipment->get_id() ) + ); + + foreach ( $items as $item ) { + wp_cache_set( 'item-' . $item->shipment_item_id, $item, 'shipment-items' ); + } + + if ( 0 < $shipment->get_id() ) { + wp_cache_set( 'shipment-items-' . $shipment->get_id(), $items, 'shipments' ); + } + } + + if ( ! empty( $items ) ) { + + $shipment_type = $shipment->get_type(); + + $items = array_map( + function( $item_id ) use ( $shipment_type ) { + return wc_gzd_get_shipment_item( $item_id, $shipment_type ); + }, + array_combine( wp_list_pluck( $items, 'shipment_item_id' ), $items ) + ); + } else { + $items = array(); + } + + return $items; + } + + /** + * Remove all items from the shipment. + * + * @param \Vendidero\Germanized\Shipments\Shipment $shipment Shipment object. + */ + public function delete_items( $shipment ) { + global $wpdb; + + $wpdb->query( $wpdb->prepare( "DELETE FROM itemmeta USING {$wpdb->gzd_shipment_itemmeta} itemmeta INNER JOIN {$wpdb->gzd_shipment_items} items WHERE itemmeta.gzd_shipment_item_id = items.shipment_item_id and items.shipment_id = %d", $shipment->get_id() ) ); + $wpdb->query( $wpdb->prepare( "DELETE FROM {$wpdb->gzd_shipment_items} WHERE shipment_id = %d", $shipment->get_id() ) ); + + $this->clear_caches( $shipment ); + } + + /** + * Get valid WP_Query args from a WC_Order_Query's query variables. + * + * @since 3.0.6 + * @param array $query_vars query vars from a WC_Order_Query. + * @return array + */ + protected function get_wp_query_args( $query_vars ) { + global $wpdb; + + // Add the 'wc-' prefix to status if needed. + if ( ! empty( $query_vars['status'] ) ) { + if ( is_array( $query_vars['status'] ) ) { + foreach ( $query_vars['status'] as &$status ) { + $status = wc_gzd_is_shipment_status( 'gzd-' . $status ) ? 'gzd-' . $status : $status; + } + } else { + $query_vars['status'] = wc_gzd_is_shipment_status( 'gzd-' . $query_vars['status'] ) ? 'gzd-' . $query_vars['status'] : $query_vars['status']; + } + } + + $wp_query_args = parent::get_wp_query_args( $query_vars ); + + // Force type to be existent + if ( isset( $query_vars['type'] ) ) { + $wp_query_args['type'] = $query_vars['type']; + } + + if ( ! isset( $wp_query_args['date_query'] ) ) { + $wp_query_args['date_query'] = array(); + } + + if ( ! isset( $wp_query_args['meta_query'] ) ) { + $wp_query_args['meta_query'] = array(); // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query + } + + // Allow Woo to treat these props as date query compatible + $date_queries = array( + 'date_created', + 'date_sent', + 'est_delivery_date', + ); + + foreach ( $date_queries as $db_key ) { + if ( isset( $query_vars[ $db_key ] ) && '' !== $query_vars[ $db_key ] ) { + + // Remove any existing meta queries for the same keys to prevent conflicts. + $existing_queries = wp_list_pluck( $wp_query_args['meta_query'], 'key', true ); + $meta_query_index = array_search( $db_key, $existing_queries, true ); + + if ( false !== $meta_query_index ) { + unset( $wp_query_args['meta_query'][ $meta_query_index ] ); + } + + $date_query_args = $this->parse_date_for_wp_query( $query_vars[ $db_key ], 'post_date', array() ); + + /** + * Replace date query columns after Woo parsed dates. + * Include table name because otherwise WP_Date_Query won't accept our custom column. + */ + if ( isset( $date_query_args['date_query'] ) && ! empty( $date_query_args['date_query'] ) ) { + $date_query = $date_query_args['date_query'][0]; + + if ( 'post_date' === $date_query['column'] ) { + $date_query['column'] = $wpdb->gzd_shipments . '.shipment_' . $db_key; + } + + $wp_query_args['date_query'][] = $date_query; + } + } + } + + if ( ! isset( $query_vars['paginate'] ) || ! $query_vars['paginate'] ) { + $wp_query_args['no_found_rows'] = true; + } + + /** + * Filter to adjust Shipments query arguments after parsing. + * + * @param array $wp_query_args Array containing parsed query arguments. + * @param array $query_vars The original query arguments. + * @param Shipment $data_store The shipment data store object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipping_data_store_get_shipments_query', $wp_query_args, $query_vars, $this ); + } + + /** + * Table structure is slightly different between meta types, this function will return what we need to know. + * + * @since 3.0.0 + * @return array Array elements: table, object_id_field, meta_id_field + */ + protected function get_db_info() { + global $wpdb; + + $meta_id_field = 'meta_id'; // for some reason users calls this umeta_id so we need to track this as well. + $table = $wpdb->gzd_shipmentmeta; + $object_id_field = $this->meta_type . '_id'; + + if ( ! empty( $this->object_id_field_for_meta ) ) { + $object_id_field = $this->object_id_field_for_meta; + } + + return array( + 'table' => $table, + 'object_id_field' => $object_id_field, + 'meta_id_field' => $meta_id_field, + ); + } + + public function get_query_args( $query_vars ) { + return $this->get_wp_query_args( $query_vars ); + } + + public function get_shipment_count( $status, $type = '' ) { + global $wpdb; + + if ( empty( $type ) ) { + $query = $wpdb->prepare( "SELECT COUNT( * ) FROM {$wpdb->gzd_shipments} WHERE shipment_status = %s", $status ); + } else { + $query = $wpdb->prepare( "SELECT COUNT( * ) FROM {$wpdb->gzd_shipments} WHERE shipment_status = %s and shipment_type = %s", $status, $type ); + } + + return absint( $wpdb->get_var( $query ) ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } +} diff --git a/packages/woocommerce-germanized-shipments/src/DataStores/ShipmentItem.php b/packages/woocommerce-germanized-shipments/src/DataStores/ShipmentItem.php new file mode 100644 index 000000000..bc1debefe --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/DataStores/ShipmentItem.php @@ -0,0 +1,354 @@ +insert( + $wpdb->gzd_shipment_items, + array( + 'shipment_id' => $item->get_shipment_id(), + 'shipment_item_quantity' => $item->get_quantity(), + 'shipment_item_order_item_id' => $item->get_order_item_id(), + 'shipment_item_product_id' => $item->get_product_id(), + 'shipment_item_parent_id' => $item->get_parent_id(), + 'shipment_item_name' => $item->get_name(), + ) + ); + + $item->set_id( $wpdb->insert_id ); + $this->save_item_data( $item ); + $item->save_meta_data(); + $item->apply_changes(); + $this->clear_cache( $item ); + + /** + * Action that indicates that a new ShipmentItem has been created in the DB. + * + * @param integer $shipment_item_id The shipment item id. + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item The shipment item object. + * @param integer $shipment_id The shipment id. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_new_shipment_item', $item->get_id(), $item, $item->get_shipment_id() ); + } + + /** + * Update a shipment item in the database. + * + * @since 3.0.0 + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item Shipment item object. + */ + public function update( &$item ) { + global $wpdb; + + $changes = $item->get_changes(); + + if ( array_intersect( $this->core_props, array_keys( $changes ) ) ) { + $wpdb->update( + $wpdb->gzd_shipment_items, + array( + 'shipment_id' => $item->get_shipment_id(), + 'shipment_item_order_item_id' => $item->get_order_item_id(), + 'shipment_item_quantity' => $item->get_quantity(), + 'shipment_item_product_id' => $item->get_product_id(), + 'shipment_item_parent_id' => $item->get_parent_id(), + 'shipment_item_name' => $item->get_name(), + ), + array( 'shipment_item_id' => $item->get_id() ) + ); + } + + $this->save_item_data( $item ); + $item->save_meta_data(); + $item->apply_changes(); + $this->clear_cache( $item ); + + /** + * Action that indicates that a ShipmentItem has been updated in the DB. + * + * @param integer $shipment_item_id The shipment item id. + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item The shipment item object. + * @param integer $shipment_id The shipment id. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_item_updated', $item->get_id(), $item, $item->get_shipment_id() ); + } + + /** + * Remove a shipment item from the database. + * + * @since 3.0.0 + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item Shipment item object. + * @param array $args Array of args to pass to the delete method. + */ + public function delete( &$item, $args = array() ) { + if ( $item->get_id() ) { + global $wpdb; + + /** + * Action that fires before deleting a ShipmentItem from the DB. + * + * @param integer $shipment_item_id The shipment item id. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_before_delete_shipment_item', $item->get_id() ); + + $wpdb->delete( $wpdb->gzd_shipment_items, array( 'shipment_item_id' => $item->get_id() ) ); + $wpdb->delete( $wpdb->gzd_shipment_itemmeta, array( 'gzd_shipment_item_id' => $item->get_id() ) ); + + /** + * Action that indicates that a ShipmentItem has been deleted from the DB. + * + * @param integer $shipment_item_id The shipment item id. + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item The shipment item object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_delete_shipment_item', $item->get_id(), $item ); + $this->clear_cache( $item ); + } + } + + /** + * Read a shipment item from the database. + * + * @since 3.0.0 + * + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item Shipment item object. + * + * @throws Exception If invalid shipment item. + */ + public function read( &$item ) { + global $wpdb; + + $item->set_defaults(); + + // Get from cache if available. + $data = wp_cache_get( 'item-' . $item->get_id(), 'shipment-items' ); + + if ( false === $data ) { + $data = $wpdb->get_row( $wpdb->prepare( "SELECT * FROM {$wpdb->gzd_shipment_items} WHERE shipment_item_id = %d LIMIT 1;", $item->get_id() ) ); + wp_cache_set( 'item-' . $item->get_id(), $data, 'shipment-items' ); + } + + if ( ! $data ) { + throw new Exception( _x( 'Invalid shipment item.', 'shipments', 'woocommerce-germanized' ) ); + } + + $item->set_props( + array( + 'shipment_id' => $data->shipment_id, + 'order_item_id' => $data->shipment_item_order_item_id, + 'quantity' => $data->shipment_item_quantity, + 'product_id' => $data->shipment_item_product_id, + 'parent_id' => $data->shipment_item_parent_id, + 'name' => $data->shipment_item_name, + ) + ); + + $this->read_item_data( $item ); + $item->read_meta_data(); + $item->set_object_read( true ); + } + + /** + * Read extra data associated with the shipment item. + * + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item Shipment item object. + * @since 3.0.0 + */ + protected function read_item_data( &$item ) { + $props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + $props[ substr( $meta_key, 1 ) ] = get_metadata( $this->meta_type, $item->get_id(), $meta_key, true ); + } + + $item->set_props( $props ); + } + + /** + * Update meta data in, or delete it from, the database. + * + * Avoids storing meta when it's either an empty string or empty array. + * Other empty values such as numeric 0 and null should still be stored. + * Data-stores can force meta to exist using `must_exist_meta_keys`. + * + * Note: WordPress `get_metadata` function returns an empty string when meta data does not exist. + * + * @param WC_Data $object The WP_Data object (WC_Coupon for coupons, etc). + * @param string $meta_key Meta key to update. + * @param mixed $meta_value Value to save. + * + * @since 3.6.0 Added to prevent empty meta being stored unless required. + * + * @return bool True if updated/deleted. + */ + protected function update_or_delete_meta( $object, $meta_key, $meta_value ) { + if ( in_array( $meta_value, array( array(), '' ), true ) && ! in_array( $meta_key, $this->must_exist_meta_keys, true ) ) { + $updated = delete_metadata( $this->meta_type, $object->get_id(), $meta_key ); + } else { + $updated = update_metadata( $this->meta_type, $object->get_id(), $meta_key, $meta_value ); + } + + return (bool) $updated; + } + + /** + * Saves an item's data to the database / item meta. + * Ran after both create and update, so $item->get_id() will be set. + * + * @since 3.0.0 + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item Shipment item object. + */ + public function save_item_data( &$item ) { + $updated_props = array(); + $meta_key_to_props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + + $prop_name = substr( $meta_key, 1 ); + + if ( in_array( $prop_name, $this->core_props, true ) ) { + continue; + } + + $meta_key_to_props[ $meta_key ] = $prop_name; + } + + $props_to_update = $this->get_props_to_update( $item, $meta_key_to_props, $this->meta_type ); + + foreach ( $props_to_update as $meta_key => $prop ) { + + $getter = "get_$prop"; + + if ( ! is_callable( array( $item, $getter ) ) ) { + continue; + } + + $value = $item->{"get_$prop"}( 'edit' ); + $value = is_string( $value ) ? wp_slash( $value ) : $value; + + $updated = $this->update_or_delete_meta( $item, $meta_key, $value ); + + if ( $updated ) { + $updated_props[] = $prop; + } + } + + /** + * Action that fires after updating a ShipmentItem's properties. + * + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item The shipment item object. + * @param array $changed_props The updated properties. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_item_object_updated_props', $item, $updated_props ); + } + + /** + * Clear meta cache. + * + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item Shipment item object. + */ + public function clear_cache( &$item ) { + wp_cache_delete( 'item-' . $item->get_id(), 'shipment-items' ); + wp_cache_delete( 'shipment-items-' . $item->get_shipment_id(), 'shipments' ); + wp_cache_delete( $item->get_id(), $this->meta_type . '_meta' ); + } + + /** + * Table structure is slightly different between meta types, this function will return what we need to know. + * + * @since 3.0.0 + * @return array Array elements: table, object_id_field, meta_id_field + */ + protected function get_db_info() { + global $wpdb; + + $meta_id_field = 'meta_id'; // for some reason users calls this umeta_id so we need to track this as well. + $table = $wpdb->gzd_shipment_itemmeta; + $object_id_field = $this->meta_type . '_id'; + + if ( ! empty( $this->object_id_field_for_meta ) ) { + $object_id_field = $this->object_id_field_for_meta; + } + + return array( + 'table' => $table, + 'object_id_field' => $object_id_field, + 'meta_id_field' => $meta_id_field, + ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/DataStores/ShippingProvider.php b/packages/woocommerce-germanized-shipments/src/DataStores/ShippingProvider.php new file mode 100644 index 000000000..73dc56928 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/DataStores/ShippingProvider.php @@ -0,0 +1,492 @@ +set_name( $this->get_unqiue_name( $provider ) ); + + if ( 0 === $provider->get_order( 'edit' ) ) { + $max_order = 1; + $max_order_col = $wpdb->get_col( "SELECT MAX(shipping_provider_order) FROM {$wpdb->gzd_shipping_provider}" ); + + if ( ! empty( $max_order_col ) ) { + $max_order = absint( $max_order_col[0] ) + 1; + } + + $provider->set_order( $max_order ); + } + + $data = array( + 'shipping_provider_activated' => $provider->is_activated() ? 1 : 0, + 'shipping_provider_name' => $provider->get_name( 'edit' ), + 'shipping_provider_title' => $provider->get_title( 'edit' ), + 'shipping_provider_order' => $provider->get_order( 'edit' ), + ); + + $wpdb->insert( + $wpdb->gzd_shipping_provider, + $data + ); + + $provider_id = $wpdb->insert_id; + + if ( $provider_id ) { + $provider->set_id( $provider_id ); + + $this->save_provider_data( $provider ); + + $provider->update_settings_with_defaults(); + $provider->save_meta_data(); + $provider->apply_changes(); + + $this->clear_caches( $provider ); + + /** + * Action that indicates that a new Shipping Provider has been created in the DB. + * + * @param integer $provider_id The provider id. + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $shipping_provider The shipping provider instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_new_shipping_provider', $provider_id, $provider ); + } + } + + /** + * Generate a unique name to save to the object. + * + * @since 3.6.0 + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider Shipping provider object. + * @return string + */ + protected function get_unqiue_name( $provider ) { + global $wpdb; + + $slug = sanitize_key( $provider->get_title() ); + + // Post slugs must be unique across all posts. + $check_sql = "SELECT shipping_provider_name FROM $wpdb->gzd_shipping_provider WHERE shipping_provider_name = %s AND shipping_provider_id != %d LIMIT 1"; + $provider_name_check = $wpdb->get_var( $wpdb->prepare( $check_sql, $slug, $provider->get_id() ) ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + + if ( $provider_name_check || ( $this->is_manual_creation_request() && $this->is_reserved_name( $slug ) ) ) { + $suffix = 2; + do { + $alt_provider_name = _truncate_post_slug( $slug, 200 - ( strlen( $suffix ) + 1 ) ) . "-$suffix"; + $provider_name_check = $wpdb->get_var( $wpdb->prepare( $check_sql, $alt_provider_name, $provider->get_id() ) ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + $suffix++; + } while ( $provider_name_check || ( $this->is_manual_creation_request() && $this->is_reserved_name( $alt_provider_name ) ) ); + $slug = $alt_provider_name; + } + + return $slug; + } + + protected function is_manual_creation_request() { + return apply_filters( 'woocommerce_gzd_shipments_shipping_provider_is_manual_creation_request', false ); + } + + protected function is_reserved_name( $name ) { + $reserved_names = array( + 'dhl', + 'deutsche_post', + 'dpd', + 'gls', + 'ups', + 'hermes', + ); + + return apply_filters( 'woocommerce_gzd_shipments_shipping_provider_is_reserved_name', in_array( $name, $reserved_names, true ) ); + } + + /** + * Method to update a shipping provider in the database. + * + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider Shipping provider object. + */ + public function update( &$provider ) { + global $wpdb; + + $updated_props = array(); + $core_props = $this->core_props; + $changed_props = array_keys( $provider->get_changes() ); + $provider_data = array(); + + foreach ( $changed_props as $prop ) { + + if ( ! in_array( $prop, $core_props, true ) ) { + continue; + } + + switch ( $prop ) { + case 'activated': + $provider_data[ 'shipping_provider_' . $prop ] = $provider->is_activated() ? 1 : 0; + break; + default: + if ( is_callable( array( $provider, 'get_' . $prop ) ) ) { + $provider_data[ 'shipping_provider_' . $prop ] = $provider->{'get_' . $prop}( 'edit' ); + } + break; + } + } + + if ( ! empty( $provider_data ) ) { + $wpdb->update( + $wpdb->gzd_shipping_provider, + $provider_data, + array( 'shipping_provider_id' => $provider->get_id() ) + ); + } + + $this->save_provider_data( $provider ); + + $provider->save_meta_data(); + $provider->apply_changes(); + + $this->clear_caches( $provider ); + + /** + * Action that indicates that a shipping provider has been updated in the DB. + * + * @param integer $shipping_provider_id The shipping provider id. + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $shipping_provider The shipping provider instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipping_provider_updated', $provider->get_id(), $provider ); + } + + /** + * Remove a shipping provider from the database. + * + * @since 3.0.0 + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider Shipping provider object. + * @param bool $force_delete Unused param. + */ + public function delete( &$provider, $force_delete = false ) { + global $wpdb; + + $wpdb->delete( $wpdb->gzd_shipping_provider, array( 'shipping_provider_id' => $provider->get_id() ), array( '%d' ) ); + $wpdb->delete( $wpdb->gzd_shipping_providermeta, array( 'gzd_shipping_provider_id' => $provider->get_id() ), array( '%d' ) ); + + $this->clear_caches( $provider ); + + /** + * Action that indicates that a shipping provider has been deleted from the DB. + * + * @param integer $shipping_provider_id The shipping provider id. + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider The shipping provider object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipping_provider_deleted', $provider->get_id(), $provider ); + } + + /** + * Read a shipping provider from the database. + * + * @since 3.0.0 + * + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider Shipping provider object. + * + * @throws Exception Throw exception if invalid shipping provider. + */ + public function read( &$provider ) { + global $wpdb; + + $data = $wpdb->get_row( + $wpdb->prepare( + "SELECT * FROM {$wpdb->gzd_shipping_provider} WHERE shipping_provider_id = %d LIMIT 1", + $provider->get_id() + ) + ); + + if ( $data ) { + $provider->set_props( + array( + 'name' => $data->shipping_provider_name, + 'title' => $data->shipping_provider_title, + 'activated' => $data->shipping_provider_activated, + 'order' => $data->shipping_provider_order, + ) + ); + + $this->read_provider_data( $provider ); + + $provider->read_meta_data(); + $provider->set_object_read( true ); + + /** + * Action that indicates that a shipping provider has been loaded from DB. + * + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider The shipping provider object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipping_provider_loaded', $provider ); + } else { + throw new Exception( _x( 'Invalid shipping provider.', 'shipments', 'woocommerce-germanized' ) ); + } + } + + public function is_activated( $name ) { + global $wpdb; + + $data = $wpdb->get_row( + $wpdb->prepare( + "SELECT shipping_provider_activated FROM {$wpdb->gzd_shipping_provider} WHERE shipping_provider_name = %s LIMIT 1", + $name + ) + ); + + if ( $data ) { + return wc_string_to_bool( $data->shipping_provider_activated ); + } + + return false; + } + + /** + * Clear any caches. + * + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider Shipping provider object. + * @since 3.0.0 + */ + protected function clear_caches( &$provider ) { + wp_cache_delete( $provider->get_id(), $this->meta_type . '_meta' ); + } + + /* + |-------------------------------------------------------------------------- + | Additional Methods + |-------------------------------------------------------------------------- + */ + + /** + * Read extra data associated with the shipping provider. + * + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider Shipping provider object. + * @since 3.0.0 + */ + protected function read_provider_data( &$provider ) { + $props = array(); + $meta_keys = $this->internal_meta_keys; + + foreach ( $provider->get_extra_data_keys() as $key ) { + $meta_keys[] = '_' . $key; + } + + foreach ( $meta_keys as $meta_key ) { + $props[ substr( $meta_key, 1 ) ] = get_metadata( 'gzd_shipping_provider', $provider->get_id(), $meta_key, true ); + } + + $provider->set_props( $props ); + } + + /** + * Save shipping provider data. + * + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider Shipping provider object. + */ + protected function save_provider_data( &$provider ) { + $updated_props = array(); + $meta_key_to_props = array(); + + foreach ( $this->internal_meta_keys as $meta_key ) { + $prop_name = substr( $meta_key, 1 ); + + if ( in_array( $prop_name, $this->core_props, true ) ) { + continue; + } + + $meta_key_to_props[ $meta_key ] = $prop_name; + } + + // Make sure to take extra data (like product url or text for external products) into account. + $extra_data_keys = $provider->get_extra_data_keys(); + + foreach ( $extra_data_keys as $key ) { + $meta_key_to_props[ '_' . $key ] = $key; + } + + $props_to_update = $this->get_props_to_update( $provider, $meta_key_to_props, 'gzd_shipping_provider' ); + + foreach ( $props_to_update as $meta_key => $prop ) { + + if ( ! is_callable( array( $provider, "get_$prop" ) ) ) { + continue; + } + + $value = $provider->{"get_$prop"}( 'edit' ); + $value = is_string( $value ) ? wp_slash( $value ) : $value; + + if ( is_bool( $value ) ) { + $value = wc_bool_to_string( $value ); + } + + $updated = $this->update_or_delete_meta( $provider, $meta_key, $value ); + + if ( $updated ) { + $updated_props[] = $prop; + } + } + + /** + * Action that fires after updating a shipping providers' properties. + * + * @param \Vendidero\Germanized\Shipments\ShippingProvider\Simple $provider The shipping provider object. + * @param array $changed_props The updated properties. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipping_provider_object_updated_props', $provider, $updated_props ); + } + + /** + * Update meta data in, or delete it from, the database. + * + * Avoids storing meta when it's either an empty string or empty array. + * Other empty values such as numeric 0 and null should still be stored. + * Data-stores can force meta to exist using `must_exist_meta_keys`. + * + * Note: WordPress `get_metadata` function returns an empty string when meta data does not exist. + * + * @param WC_Data $object The WP_Data object (WC_Coupon for coupons, etc). + * @param string $meta_key Meta key to update. + * @param mixed $meta_value Value to save. + * + * @since 3.6.0 Added to prevent empty meta being stored unless required. + * + * @return bool True if updated/deleted. + */ + protected function update_or_delete_meta( $object, $meta_key, $meta_value ) { + if ( in_array( $meta_value, array( array(), '' ), true ) && ! in_array( $meta_key, $this->must_exist_meta_keys, true ) ) { + $updated = delete_metadata( 'gzd_shipping_provider', $object->get_id(), $meta_key ); + } else { + $updated = update_metadata( 'gzd_shipping_provider', $object->get_id(), $meta_key, $meta_value ); + } + + return (bool) $updated; + } + + /** + * Table structure is slightly different between meta types, this function will return what we need to know. + * + * @since 3.0.0 + * @return array Array elements: table, object_id_field, meta_id_field + */ + protected function get_db_info() { + global $wpdb; + + $meta_id_field = 'meta_id'; // for some reason users calls this umeta_id so we need to track this as well. + $table = $wpdb->gzd_shipping_providermeta; + $object_id_field = $this->meta_type . '_id'; + + if ( ! empty( $this->object_id_field_for_meta ) ) { + $object_id_field = $this->object_id_field_for_meta; + } + + return array( + 'table' => $table, + 'object_id_field' => $object_id_field, + 'meta_id_field' => $meta_id_field, + ); + } + + public function get_shipping_provider_count() { + global $wpdb; + + return absint( $wpdb->get_var( "SELECT COUNT( * ) FROM {$wpdb->gzd_shipping_provider}" ) ); + } + + public function get_shipping_provider_name( $provider_id ) { + global $wpdb; + + $provider_name_check = $wpdb->get_row( $wpdb->prepare( "SELECT shipping_provider_name FROM $wpdb->gzd_shipping_provider WHERE shipping_provider_id = %d LIMIT 1", $provider_id ) ); + + if ( ! empty( $provider_name_check ) ) { + return $provider_name_check->shipping_provider_name; + } + + return false; + } + + public function get_shipping_providers() { + global $wpdb; + + $providers = $wpdb->get_results( "SELECT * FROM $wpdb->gzd_shipping_provider ORDER BY shipping_provider_order ASC" ); + $shipping_providers = array(); + + foreach ( $providers as $provider ) { + try { + $shipping_providers[ $provider->shipping_provider_name ] = $provider; + } catch ( Exception $e ) { + continue; + } + } + + return $shipping_providers; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Emails.php b/packages/woocommerce-germanized-shipments/src/Emails.php new file mode 100644 index 000000000..a0c772f1c --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Emails.php @@ -0,0 +1,232 @@ +id ), $email ) ) ) { + if ( $shipments_order = wc_gzd_get_shipment_order( $order ) ) { + $template = $plain_text ? 'plain/email-order-shipments.php' : 'email-order-shipments.php'; + + wc_get_template( + 'emails/' . $template, + array( + 'shipments' => $shipments_order->get_simple_shipments( true ), + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } + } + } + + public static function set_woocommerce_template_dir( $dir, $template ) { + if ( file_exists( Package::get_path() . '/templates/' . $template ) ) { + return 'woocommerce-germanized'; + } + + return $dir; + } + + public static function register_emails( $emails ) { + $emails['WC_GZD_Email_Customer_Shipment'] = include Package::get_path() . '/includes/emails/class-wc-gzd-email-customer-shipment.php'; + $emails['WC_GZD_Email_Customer_Return_Shipment'] = include Package::get_path() . '/includes/emails/class-wc-gzd-email-customer-return-shipment.php'; + $emails['WC_GZD_Email_Customer_Return_Shipment_Delivered'] = include Package::get_path() . '/includes/emails/class-wc-gzd-email-customer-return-shipment-delivered.php'; + $emails['WC_GZD_Email_Customer_Guest_Return_Shipment_Request'] = include Package::get_path() . '/includes/emails/class-wc-gzd-email-customer-guest-return-shipment-request.php'; + $emails['WC_GZD_Email_New_Return_Shipment_Request'] = include Package::get_path() . '/includes/emails/class-wc-gzd-email-new-return-shipment-request.php'; + + return $emails; + } + + public static function email_hooks() { + add_action( 'woocommerce_gzd_email_shipment_details', array( __CLASS__, 'email_return_instructions' ), 5, 4 ); + add_action( 'woocommerce_gzd_email_shipment_details', array( __CLASS__, 'email_tracking' ), 10, 4 ); + add_action( 'woocommerce_gzd_email_shipment_details', array( __CLASS__, 'email_address' ), 20, 4 ); + add_action( 'woocommerce_gzd_email_shipment_details', array( __CLASS__, 'email_details' ), 30, 4 ); + } + + public static function register_email_notifications( $actions ) { + + $actions = array_merge( + $actions, + array( + 'woocommerce_gzd_shipment_status_draft_to_processing', + 'woocommerce_gzd_shipment_status_draft_to_shipped', + 'woocommerce_gzd_shipment_status_draft_to_delivered', + 'woocommerce_gzd_shipment_status_processing_to_shipped', + 'woocommerce_gzd_shipment_status_processing_to_delivered', + 'woocommerce_gzd_shipment_status_shipped_to_delivered', + 'woocommerce_gzd_return_shipment_status_draft_to_processing', + 'woocommerce_gzd_return_shipment_status_draft_to_shipped', + 'woocommerce_gzd_return_shipment_status_draft_to_delivered', + 'woocommerce_gzd_return_shipment_status_draft_to_requested', + 'woocommerce_gzd_return_shipment_status_processing_to_shipped', + 'woocommerce_gzd_return_shipment_status_processing_to_delivered', + 'woocommerce_gzd_return_shipment_status_shipped_to_delivered', + 'woocommerce_gzd_return_shipment_status_requested_to_processing', + 'woocommerce_gzd_return_shipment_status_requested_to_shipped', + ) + ); + + return $actions; + } + + /** + * @param Shipment $shipment + * @param bool $sent_to_admin + * @param bool $plain_text + * @param string $email + */ + public static function email_return_instructions( $shipment, $sent_to_admin = false, $plain_text = false, $email = '' ) { + + if ( 'return' !== $shipment->get_type() || $shipment->has_status( 'delivered' ) ) { + return; + } + + if ( $plain_text ) { + wc_get_template( + 'emails/plain/email-return-shipment-instructions.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } else { + wc_get_template( + 'emails/email-return-shipment-instructions.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } + } + + /** + * @param Shipment $shipment + * @param bool $sent_to_admin + * @param bool $plain_text + * @param string $email + */ + public static function email_tracking( $shipment, $sent_to_admin = false, $plain_text = false, $email = '' ) { + + // Do only include shipment tracking if estimated delivery date or tracking instruction or tracking url exists + // Do not show tracking for returns + if ( ! $shipment->has_tracking() || $shipment->has_status( 'delivered' ) || 'return' === $shipment->get_type() ) { + return; + } + + if ( $plain_text ) { + wc_get_template( + 'emails/plain/email-shipment-tracking.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } else { + wc_get_template( + 'emails/email-shipment-tracking.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } + } + + /** + * @param Shipment $shipment + * @param bool $sent_to_admin + * @param bool $plain_text + * @param string $email + */ + public static function email_address( $shipment, $sent_to_admin = false, $plain_text = false, $email = '' ) { + if ( 'return' === $shipment->get_type() || in_array( $email, array( 'customer_return_shipment_delivered' ), true ) ) { + if ( $provider = $shipment->get_shipping_provider_instance() ) { + if ( $provider->hide_return_address() ) { + return; + } + } + } + + if ( $plain_text ) { + wc_get_template( + 'emails/plain/email-shipment-address.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } else { + wc_get_template( + 'emails/email-shipment-address.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } + } + + /** + * Show the order details table + * + * @param \WC_Order $order Order instance. + * @param bool $sent_to_admin If should sent to admin. + * @param bool $plain_text If is plain text email. + * @param string $email Email address. + */ + public static function email_details( $shipment, $sent_to_admin = false, $plain_text = false, $email = '' ) { + if ( $plain_text ) { + wc_get_template( + 'emails/plain/email-shipment-details.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } else { + wc_get_template( + 'emails/email-shipment-details.php', + array( + 'shipment' => $shipment, + 'sent_to_admin' => $sent_to_admin, + 'plain_text' => $plain_text, + 'email' => $email, + ) + ); + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/FormHandler.php b/packages/woocommerce-germanized-shipments/src/FormHandler.php new file mode 100644 index 000000000..75eeab9a6 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/FormHandler.php @@ -0,0 +1,339 @@ +' . _x( 'Error:', 'shipments', 'woocommerce-germanized' ) . ' ' . _x( 'We were not able to find a matching order.', 'shipments', 'woocommerce-germanized' ) ); + } + + if ( ! wc_gzd_order_is_customer_returnable( $order ) ) { + throw new Exception( '' . _x( 'Error:', 'shipments', 'woocommerce-germanized' ) . ' ' . _x( 'This order is currently not eligible for returns. Please contact us for further details.', 'shipments', 'woocommerce-germanized' ) ); + } + + $key = 'wc_gzd_order_return_request_' . wp_generate_password( 13, false ); + + $order->update_meta_data( '_return_request_key', $key ); + $order->save(); + + // Send email to customer + wc_add_notice( _x( 'Thank you. You\'ll receive an email containing a link to create a new return to your order.', 'shipments', 'woocommerce-germanized' ), 'success' ); + + WC()->mailer()->emails['WC_GZD_Email_Customer_Guest_Return_Shipment_Request']->trigger( $order ); + + do_action( 'woocommerce_gzd_return_request_successfull', $order ); + + } catch ( Exception $e ) { + wc_add_notice( $e->getMessage(), 'error' ); + do_action( 'woocommerce_gzd_return_request_failed' ); + } + } + } + + /** + * @param $order_id + * @param $email + * + * @return false|integer + */ + public static function find_order( $order_id, $email ) { + $order_id_parsed = self::get_order_id_from_string( $order_id ); + $db_order_id = false; + $orders = wc_get_orders( + apply_filters( + 'woocommerce_gzd_return_request_order_query_args', + array( + 'billing_email' => $email, + 'post__in' => array( $order_id_parsed ), + 'limit' => 1, + 'return' => 'ids', + ) + ) + ); + + // Now lets try to find the order by a custom order number field + if ( empty( $orders ) ) { + add_filter( 'woocommerce_order_data_store_cpt_get_orders_query', array( __CLASS__, 'filter_query_by_order_number' ), 10, 2 ); + + $orders = wc_get_orders( + apply_filters( + 'woocommerce_gzd_return_request_alternate_order_query_args', + array( + 'billing_email' => $email, + 'order_number' => $order_id, + 'limit' => 1, + 'return' => 'ids', + ) + ) + ); + + remove_filter( 'woocommerce_order_data_store_cpt_get_orders_query', array( __CLASS__, 'filter_query_by_order_number' ), 10 ); + } + + if ( ! empty( $orders ) ) { + $db_order_id = $orders[0]; + } + + return apply_filters( 'woocommerce_gzd_shipments_valid_order_for_return_request', $db_order_id, $order_id, $email ); + } + + public static function filter_query_by_order_number( $query, $query_vars ) { + $meta_field_name = apply_filters( 'woocommerce_gzd_return_request_customer_order_number_meta_key', '_order_number' ); + + if ( ! empty( $query_vars['order_number'] ) ) { + $query['meta_query'][] = array( + 'key' => $meta_field_name, + 'value' => esc_attr( wc_clean( $query_vars['order_number'] ) ), + 'compare' => '=', + ); + } + + return $query; + } + + /** + * Save the password/account details and redirect back to the my account page. + */ + public static function add_return_shipment() { + $nonce_value = isset( $_REQUEST['add-return-shipment-nonce'] ) ? wp_unslash( $_REQUEST['add-return-shipment-nonce'] ) : ''; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + + if ( ! wp_verify_nonce( $nonce_value, 'add_return_shipment' ) ) { + return; + } + + if ( empty( $_POST['action'] ) || 'gzd_add_return_shipment' !== $_POST['action'] ) { + return; + } + + wc_nocache_headers(); + + $order_id = ! empty( $_POST['order_id'] ) ? absint( wp_unslash( $_POST['order_id'] ) ) : false; + $items = ! empty( $_POST['items'] ) ? wc_clean( wp_unslash( $_POST['items'] ) ) : array(); + $item_data = ! empty( $_POST['item'] ) ? wc_clean( wp_unslash( $_POST['item'] ) ) : array(); + + if ( ! ( $order = wc_get_order( $order_id ) ) || ( ! wc_gzd_customer_can_add_return_shipment( $order_id ) ) ) { + wc_add_notice( _x( 'You are not allowed to add returns to that order.', 'shipments', 'woocommerce-germanized' ), 'error' ); + return; + } + + if ( ! wc_gzd_order_is_customer_returnable( $order ) ) { + wc_add_notice( _x( 'Sorry, but this order does not support returns any longer.', 'shipments', 'woocommerce-germanized' ), 'error' ); + return; + } + + if ( empty( $items ) ) { + wc_add_notice( _x( 'Please choose one or more items from the list.', 'shipments', 'woocommerce-germanized' ), 'error' ); + return; + } + + $return_items = array(); + $shipment_order = wc_gzd_get_shipment_order( $order ); + + foreach ( $items as $order_item_id ) { + + if ( $item = $shipment_order->get_simple_shipment_item( $order_item_id ) ) { + + $quantity = isset( $item_data[ $order_item_id ]['quantity'] ) ? absint( $item_data[ $order_item_id ]['quantity'] ) : 0; + $quantity_returnable = $shipment_order->get_item_quantity_left_for_returning( $order_item_id ); + $reason = isset( $item_data[ $order_item_id ]['reason'] ) ? wc_clean( $item_data[ $order_item_id ]['reason'] ) : ''; + + if ( ! empty( $reason ) && ! wc_gzd_return_shipment_reason_exists( $reason ) ) { + wc_add_notice( _x( 'The return reason you have chosen does not exist.', 'shipments', 'woocommerce-germanized' ), 'error' ); + return; + } elseif ( empty( $reason ) && ! wc_gzd_allow_customer_return_empty_return_reason( $order ) ) { + wc_add_notice( _x( 'Please choose a return reason from the list.', 'shipments', 'woocommerce-germanized' ), 'error' ); + return; + } + + if ( $quantity > $quantity_returnable ) { + wc_add_notice( _x( 'Please check your item quantities. Quantities must not exceed maximum quantities.', 'shipments', 'woocommerce-germanized' ), 'error' ); + return; + } else { + $return_items[ $order_item_id ] = array( + 'quantity' => $quantity, + 'return_reason_code' => $reason, + ); + } + } + } + + if ( empty( $return_items ) ) { + wc_add_notice( _x( 'Please choose one or more items from the list.', 'shipments', 'woocommerce-germanized' ), 'error' ); + } + + if ( wc_notice_count( 'error' ) > 0 ) { + return; + } + + $needs_manual_confirmation = wc_gzd_customer_return_needs_manual_confirmation( $order ); + + if ( $needs_manual_confirmation ) { + $default_status = 'requested'; + } else { + $default_status = 'processing'; + } + + // Add return shipment + $return_shipment = wc_gzd_create_return_shipment( + $shipment_order, + array( + 'items' => $return_items, + 'props' => array( + /** + * This filter may be used to adjust the default status of a return shipment + * added by a customer. + * + * @param string $status The default status. + * @param WC_Order $order The order object. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + 'status' => apply_filters( 'woocommerce_gzd_customer_new_return_shipment_request_status', $default_status, $order ), + 'is_customer_requested' => true, + ), + ) + ); + + if ( is_wp_error( $return_shipment ) ) { + wc_add_notice( _x( 'There was an error while creating the return. Please contact us for further information.', 'shipments', 'woocommerce-germanized' ), 'error' ); + return; + } else { + // Delete return request key if available + $shipment_order->delete_order_return_request_key(); + + $success_message = self::get_return_request_success_message( $needs_manual_confirmation ); + + // Do not add success message for guest returns + if ( $order->get_customer_id() > 0 ) { + wc_add_notice( $success_message ); + } + + /** + * This hook is fired after a customer has added a new return request + * for a specific shipment. The return shipment object has been added successfully. + * + * @param ReturnShipment $shipment The return shipment object. + * @param WC_Order $order The order object. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_new_customer_return_shipment_request', $return_shipment, $order ); + + if ( $needs_manual_confirmation ) { + $return_url = $order->get_view_order_url(); + } else { + $return_url = $return_shipment->get_view_shipment_url(); + } + + if ( $order->get_customer_id() <= 0 ) { + $return_url = add_query_arg( + array( + 'return-request-success' => 'yes', + 'needs-confirmation' => wc_bool_to_string( $needs_manual_confirmation ), + ), + wc_get_page_permalink( 'myaccount' ) + ); + } + + /** + * This filter may be used to adjust the redirect of a customer + * after adding a new return shipment. In case the return request needs manual confirmation + * the customer will be redirected to the parent shipment. + * + * @param string $url The redirect URL. + * @param ReturnShipment $shipment The return shipment object. + * @param boolean $needs_manual_confirmation Whether the request needs manual confirmation or not. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + $redirect = apply_filters( 'woocommerce_gzd_customer_new_return_shipment_request_redirect', $return_url, $return_shipment, $needs_manual_confirmation ); + + wp_safe_redirect( esc_url_raw( $redirect ) ); + exit; + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Install.php b/packages/woocommerce-germanized-shipments/src/Install.php new file mode 100644 index 000000000..30e90ff37 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Install.php @@ -0,0 +1,323 @@ +get_shipping_providers(); + + foreach ( $providers as $provider ) { + if ( ! $provider->is_activated() ) { + continue; + } + + $provider->update_settings_with_defaults(); + $provider->save(); + } + } + + private static function maybe_create_return_reasons() { + $reasons = get_option( 'woocommerce_gzd_shipments_return_reasons', null ); + + if ( is_null( $reasons ) ) { + $default_reasons = array( + array( + 'order' => 1, + 'code' => 'wrong-product', + 'reason' => _x( 'Wrong product or size ordered', 'shipments', 'woocommerce-germanized' ), + ), + array( + 'order' => 2, + 'code' => 'not-needed', + 'reason' => _x( 'Product no longer needed', 'shipments', 'woocommerce-germanized' ), + ), + array( + 'order' => 3, + 'code' => 'look', + 'reason' => _x( 'Don\'t like the look', 'shipments', 'woocommerce-germanized' ), + ), + ); + + update_option( 'woocommerce_gzd_shipments_return_reasons', $default_reasons ); + } + } + + private static function get_db_version() { + return get_option( 'woocommerce_gzd_shipments_db_version', null ); + } + + private static function maybe_create_packaging() { + $packaging = wc_gzd_get_packaging_list(); + $db_version = self::get_db_version(); + + if ( empty( $packaging ) && is_null( $db_version ) ) { + $defaults = array( + array( + 'description' => _x( 'Cardboard S', 'shipments', 'woocommerce-germanized' ), + 'length' => 25, + 'width' => 17.5, + 'height' => 10, + 'weight' => 0.14, + 'max_content_weight' => 30, + 'type' => 'cardboard', + ), + array( + 'description' => _x( 'Cardboard M', 'shipments', 'woocommerce-germanized' ), + 'length' => 37.5, + 'width' => 30, + 'height' => 13.5, + 'weight' => 0.23, + 'max_content_weight' => 30, + 'type' => 'cardboard', + ), + array( + 'description' => _x( 'Cardboard L', 'shipments', 'woocommerce-germanized' ), + 'length' => 45, + 'width' => 35, + 'height' => 20, + 'weight' => 0.3, + 'max_content_weight' => 30, + 'type' => 'cardboard', + ), + array( + 'description' => _x( 'Letter C5/6', 'shipments', 'woocommerce-germanized' ), + 'length' => 22, + 'width' => 11, + 'height' => 1, + 'weight' => 0, + 'max_content_weight' => 0.05, + 'type' => 'letter', + ), + array( + 'description' => _x( 'Letter C4', 'shipments', 'woocommerce-germanized' ), + 'length' => 22.9, + 'width' => 32.4, + 'height' => 2, + 'weight' => 0.01, + 'max_content_weight' => 1, + 'type' => 'letter', + ), + ); + + foreach ( $defaults as $default ) { + $packaging = new Packaging(); + $packaging->set_props( $default ); + $packaging->save(); + } + } + } + + private static function create_tables() { + global $wpdb; + + $current_version = get_option( 'woocommerce_gzd_shipments_version', null ); + + /** + * Make possible duplicate names unique. + */ + if ( null !== $current_version && isset( $wpdb->gzd_shipping_provider ) ) { + $providers = $wpdb->get_results( "SELECT * FROM $wpdb->gzd_shipping_provider" ); + $shipping_providers = array(); + + foreach ( $providers as $provider ) { + if ( in_array( $provider->shipping_provider_name, $shipping_providers, true ) ) { + $unique_provider_name = sanitize_title( $provider->shipping_provider_name . '_' . wp_generate_password( 4, false, false ) ); + + $wpdb->update( + $wpdb->gzd_shipping_provider, + array( + 'shipping_provider_name' => $unique_provider_name, + ), + array( 'shipping_provider_id' => $provider->shipping_provider_id ) + ); + } else { + $shipping_providers[] = $provider->shipping_provider_name; + } + } + } + + $wpdb->hide_errors(); + require_once ABSPATH . 'wp-admin/includes/upgrade.php'; + dbDelta( self::get_schema() ); + } + + private static function create_upload_dir() { + Package::maybe_set_upload_dir(); + + $dir = Package::get_upload_dir(); + + if ( ! @is_dir( $dir['basedir'] ) ) { // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged + @mkdir( $dir['basedir'] ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged + } + + if ( ! file_exists( trailingslashit( $dir['basedir'] ) . '.htaccess' ) ) { + @file_put_contents( trailingslashit( $dir['basedir'] ) . '.htaccess', 'deny from all' ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged,WordPress.WP.AlternativeFunctions.file_system_read_file_put_contents + } + + if ( ! file_exists( trailingslashit( $dir['basedir'] ) . 'index.php' ) ) { + @touch( trailingslashit( $dir['basedir'] ) . 'index.php' ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged + } + } + + private static function get_schema() { + global $wpdb; + + $collate = ''; + + if ( $wpdb->has_cap( 'collation' ) ) { + $collate = $wpdb->get_charset_collate(); + } + + /** + * Use a varchar(191) for shipping_provider_name as the key length might overflow max key length for older MySQL (< 5.7). + * @see https://stackoverflow.com/a/31474509 + */ + $tables = " +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipment_items ( + shipment_item_id BIGINT UNSIGNED NOT NULL auto_increment, + shipment_id BIGINT UNSIGNED NOT NULL, + shipment_item_name TEXT NOT NULL, + shipment_item_order_item_id BIGINT UNSIGNED NOT NULL, + shipment_item_product_id BIGINT UNSIGNED NOT NULL, + shipment_item_parent_id BIGINT UNSIGNED NOT NULL, + shipment_item_quantity SMALLINT UNSIGNED NOT NULL DEFAULT '1', + PRIMARY KEY (shipment_item_id), + KEY shipment_id (shipment_id), + KEY shipment_item_order_item_id (shipment_item_order_item_id), + KEY shipment_item_product_id (shipment_item_product_id), + KEY shipment_item_parent_id (shipment_item_parent_id) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipment_itemmeta ( + meta_id BIGINT UNSIGNED NOT NULL auto_increment, + gzd_shipment_item_id BIGINT UNSIGNED NOT NULL, + meta_key varchar(255) default NULL, + meta_value longtext NULL, + PRIMARY KEY (meta_id), + KEY gzd_shipment_item_id (gzd_shipment_item_id), + KEY meta_key (meta_key(32)) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipments ( + shipment_id BIGINT UNSIGNED NOT NULL auto_increment, + shipment_date_created datetime NOT NULL default '0000-00-00 00:00:00', + shipment_date_created_gmt datetime NOT NULL default '0000-00-00 00:00:00', + shipment_date_sent datetime NOT NULL default '0000-00-00 00:00:00', + shipment_date_sent_gmt datetime NOT NULL default '0000-00-00 00:00:00', + shipment_est_delivery_date datetime NOT NULL default '0000-00-00 00:00:00', + shipment_est_delivery_date_gmt datetime NOT NULL default '0000-00-00 00:00:00', + shipment_status varchar(20) NOT NULL default 'gzd-draft', + shipment_order_id BIGINT UNSIGNED NOT NULL DEFAULT 0, + shipment_packaging_id BIGINT UNSIGNED NOT NULL DEFAULT 0, + shipment_parent_id BIGINT UNSIGNED NOT NULL DEFAULT 0, + shipment_country varchar(2) NOT NULL DEFAULT '', + shipment_tracking_id varchar(200) NOT NULL DEFAULT '', + shipment_type varchar(200) NOT NULL DEFAULT '', + shipment_version varchar(200) NOT NULL DEFAULT '', + shipment_search_index longtext NOT NULL DEFAULT '', + shipment_shipping_provider varchar(200) NOT NULL DEFAULT '', + shipment_shipping_method varchar(200) NOT NULL DEFAULT '', + PRIMARY KEY (shipment_id), + KEY shipment_order_id (shipment_order_id), + KEY shipment_packaging_id (shipment_packaging_id), + KEY shipment_parent_id (shipment_parent_id) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipment_labels ( + label_id BIGINT UNSIGNED NOT NULL auto_increment, + label_date_created datetime NOT NULL default '0000-00-00 00:00:00', + label_date_created_gmt datetime NOT NULL default '0000-00-00 00:00:00', + label_shipment_id BIGINT UNSIGNED NOT NULL, + label_parent_id BIGINT UNSIGNED NOT NULL DEFAULT 0, + label_number varchar(200) NOT NULL DEFAULT '', + label_product_id varchar(200) NOT NULL DEFAULT '', + label_shipping_provider varchar(200) NOT NULL DEFAULT '', + label_path varchar(200) NOT NULL DEFAULT '', + label_type varchar(200) NOT NULL DEFAULT '', + PRIMARY KEY (label_id), + KEY label_shipment_id (label_shipment_id), + KEY label_parent_id (label_parent_id) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipment_labelmeta ( + meta_id BIGINT UNSIGNED NOT NULL auto_increment, + gzd_shipment_label_id BIGINT UNSIGNED NOT NULL, + meta_key varchar(255) default NULL, + meta_value longtext NULL, + PRIMARY KEY (meta_id), + KEY gzd_shipment_label_id (gzd_shipment_label_id), + KEY meta_key (meta_key(32)) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipmentmeta ( + meta_id BIGINT UNSIGNED NOT NULL auto_increment, + gzd_shipment_id BIGINT UNSIGNED NOT NULL, + meta_key varchar(255) default NULL, + meta_value longtext NULL, + PRIMARY KEY (meta_id), + KEY gzd_shipment_id (gzd_shipment_id), + KEY meta_key (meta_key(32)) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_packaging ( + packaging_id BIGINT UNSIGNED NOT NULL auto_increment, + packaging_date_created datetime NOT NULL default '0000-00-00 00:00:00', + packaging_date_created_gmt datetime NOT NULL default '0000-00-00 00:00:00', + packaging_type varchar(200) NOT NULL DEFAULT '', + packaging_description TINYTEXT NOT NULL DEFAULT '', + packaging_weight DECIMAL(6,2) UNSIGNED NOT NULL DEFAULT 0, + packaging_order BIGINT UNSIGNED NOT NULL DEFAULT 0, + packaging_max_content_weight DECIMAL(6,2) UNSIGNED NOT NULL DEFAULT 0, + packaging_length DECIMAL(6,2) UNSIGNED NOT NULL DEFAULT 0, + packaging_width DECIMAL(6,2) UNSIGNED NOT NULL DEFAULT 0, + packaging_height DECIMAL(6,2) UNSIGNED NOT NULL DEFAULT 0, + PRIMARY KEY (packaging_id) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_packagingmeta ( + meta_id BIGINT UNSIGNED NOT NULL auto_increment, + gzd_packaging_id BIGINT UNSIGNED NOT NULL, + meta_key varchar(255) default NULL, + meta_value longtext NULL, + PRIMARY KEY (meta_id), + KEY gzd_packaging_id (gzd_packaging_id), + KEY meta_key (meta_key(32)) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipping_provider ( + shipping_provider_id BIGINT UNSIGNED NOT NULL auto_increment, + shipping_provider_activated TINYINT(1) NOT NULL default 1, + shipping_provider_order smallint(10) NOT NULL DEFAULT 0, + shipping_provider_title varchar(200) NOT NULL DEFAULT '', + shipping_provider_name varchar(191) NOT NULL DEFAULT '', + PRIMARY KEY (shipping_provider_id), + UNIQUE KEY shipping_provider_name (shipping_provider_name) +) $collate; +CREATE TABLE {$wpdb->prefix}woocommerce_gzd_shipping_providermeta ( + meta_id BIGINT UNSIGNED NOT NULL auto_increment, + gzd_shipping_provider_id BIGINT UNSIGNED NOT NULL, + meta_key varchar(255) default NULL, + meta_value longtext NULL, + PRIMARY KEY (meta_id), + KEY gzd_shipping_provider_id (gzd_shipping_provider_id), + KEY meta_key (meta_key(32)) +) $collate;"; + + return $tables; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Interfaces/ShipmentLabel.php b/packages/woocommerce-germanized-shipments/src/Interfaces/ShipmentLabel.php new file mode 100644 index 000000000..d1512fcf2 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Interfaces/ShipmentLabel.php @@ -0,0 +1,99 @@ +get_shipping_provider_instance() ) { + if ( $provider->automatically_set_shipment_status_shipped( $shipment ) ) { + $shipment->set_status( 'shipped' ); + } + } + } + + public static function set_after_create_automation( $shipment_id, $shipment ) { + self::do_automation( $shipment, false ); + } + + /** + * @param Shipment $shipment + * @param boolean $is_hook + */ + protected static function do_automation( $shipment, $is_hook = true ) { + $disable = false; + + if ( ! $shipment->needs_label( false ) ) { + $disable = true; + } + + /** + * Filter that allows to disable automatically creating DHL labels for a certain shipment. + * + * @param boolean $disable True if you want to disable automation. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + $disable = apply_filters( 'woocommerce_gzd_shipments_disable_label_auto_generate', $disable, $shipment ); + + if ( $disable ) { + return; + } + + if ( $provider = $shipment->get_shipping_provider_instance() ) { + $hook_prefix = 'woocommerce_gzd_' . ( 'return' === $shipment->get_type() ? 'return_' : '' ) . 'shipment_status_'; + $auto_status = $provider->get_label_automation_shipment_status( $shipment ); + + if ( $provider->automatically_generate_label( $shipment ) && ! empty( $auto_status ) ) { + $status = str_replace( 'gzd-', '', $auto_status ); + + if ( $is_hook ) { + add_action( $hook_prefix . $status, array( __CLASS__, 'maybe_create_label' ), 5, 2 ); + } elseif ( $shipment->has_status( $status ) ) { + self::maybe_create_label( $shipment->get_id(), $shipment ); + } + } + } + } + + /** + * @param $shipment_id + * @param Shipment $shipment + */ + public static function set_automation( $shipment_id, $shipment ) { + self::do_automation( $shipment, true ); + } + + public static function create_label( $shipment_id, $shipment = false ) { + if ( ! $shipment ) { + $shipment = wc_gzd_get_shipment( $shipment_id ); + + if ( ! $shipment ) { + return; + } + } + + if ( ! $shipment->has_label() ) { + $result = $shipment->create_label(); + + if ( is_wp_error( $result ) ) { + $result = wc_gzd_get_shipment_error( $result ); + } + + if ( is_wp_error( $result ) ) { + if ( $result->is_soft_error() ) { + Package::log( sprintf( 'Info while automatically creating label for %1$s: %2$s', $shipment->get_shipment_number(), wc_print_r( $result->get_error_messages(), true ) ) ); + } else { + Package::log( sprintf( 'Error while automatically creating label for %1$s: %2$s', $shipment->get_shipment_number(), wc_print_r( $result->get_error_messages(), true ) ) ); + } + } + } + } + + private static function is_admin_edit_order_request() { + return ( isset( $_POST['action'] ) && 'editpost' === $_POST['action'] && isset( $_POST['post_type'] ) && 'shop_order' === $_POST['post_type'] ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + public static function maybe_create_label( $shipment_id, $shipment = false ) { + // Make sure that MetaBox is saved before we process automation + if ( self::is_admin_edit_order_request() && ! did_action( 'woocommerce_process_shop_order_meta' ) ) { + add_action( 'woocommerce_process_shop_order_meta', array( __CLASS__, 'create_label' ), 70 ); + } else { + self::create_label( $shipment_id, $shipment ); + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Labels/DownloadHandler.php b/packages/woocommerce-germanized-shipments/src/Labels/DownloadHandler.php new file mode 100644 index 000000000..a99679a11 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Labels/DownloadHandler.php @@ -0,0 +1,133 @@ + isset( $_GET['force'] ) ? wc_clean( wp_unslash( $_GET['force'] ) ) : 'no', // phpcs:ignore WordPress.Security.NonceVerification.Recommended + 'path' => isset( $_GET['print'] ) ? wc_clean( wp_unslash( $_GET['print'] ) ) : 'no', // phpcs:ignore WordPress.Security.NonceVerification.Recommended + ); + + $args = self::parse_args( $args ); + + if ( current_user_can( 'edit_shop_orders' ) ) { + $handler = new BulkLabel(); + + if ( $path = $handler->get_file() ) { + $filename = $handler->get_filename(); + + self::download( $path, $filename, $args['force'] ); + } + } + } + } + } + + public static function download_label() { + if ( 'wc-gzd-download-shipment-label' === $_GET['action'] && wp_verify_nonce( wp_unslash( $_REQUEST['_wpnonce'] ), 'download-shipment-label' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotValidated,WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + $shipment_id = absint( $_GET['shipment_id'] ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotValidated + $has_permission = current_user_can( 'edit_shop_orders' ); + + $args = self::parse_args( + array( + 'force' => wc_string_to_bool( isset( $_GET['force'] ) ? wc_clean( wp_unslash( $_GET['force'] ) ) : false ), + ) + ); + + if ( $shipment = wc_gzd_get_shipment( $shipment_id ) ) { + if ( 'return' === $shipment->get_type() && current_user_can( 'view_order', $shipment->get_order_id() ) && $shipment->has_label() ) { + $has_permission = true; + } + + if ( $has_permission ) { + if ( $label = $shipment->get_label() ) { + $file = $label->get_file( $args['path'] ); + $filename = $label->get_filename( $args['path'] ); + + if ( file_exists( $file ) ) { + self::download( $file, $filename, $args['force'] ); + } + } + } + } + } + } + + public static function parse_args( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'force' => false, + 'path' => '', + ) + ); + + $args['force'] = wc_string_to_bool( $args['force'] ); + + return $args; + } + + public static function download( $path, $filename, $force = false ) { + if ( $force ) { + self::force( $path, $filename ); + } else { + self::embed( $path, $filename ); + } + } + + private static function force( $path, $filename ) { + WC_Download_Handler::download_file_force( $path, $filename ); + } + + private static function embed( $file_path, $filename ) { + if ( ob_get_level() ) { + $levels = ob_get_level(); + for ( $i = 0; $i < $levels; $i++ ) { + @ob_end_clean(); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged + } + } else { + @ob_end_clean(); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged + } + + wc_nocache_headers(); + + header( 'X-Robots-Tag: noindex, nofollow', true ); + header( 'Content-type: application/pdf' ); + header( 'Content-Description: File Transfer' ); + header( 'Content-Disposition: inline; filename="' . $filename . '";' ); + header( 'Content-Transfer-Encoding: binary' ); + + $file_size = @filesize( $file_path ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged + + if ( ! $file_size ) { + return; + } + + header( 'Content-Length: ' . $file_size ); + + @readfile( $file_path ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged,WordPress.WP.AlternativeFunctions.file_system_read_readfile + exit(); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Labels/Factory.php b/packages/woocommerce-germanized-shipments/src/Labels/Factory.php new file mode 100644 index 000000000..49d52f456 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Labels/Factory.php @@ -0,0 +1,91 @@ +get_label_data( $label_id ); + + if ( $label_data ) { + $label_type = $label_data->type; + $shipping_provider_name = $label_data->shipping_provider; + } + } + + $shipping_provider_name = apply_filters( 'woocommerce_gzd_shipment_label_shipping_provider_name', $shipping_provider_name, $label_id, $label_type ); + + if ( ! $shipping_provider = wc_gzd_get_shipping_provider( $shipping_provider_name ) ) { + return false; + } + + /** + * Simple shipping provider do not support labels + */ + if ( ! is_a( $shipping_provider, '\Vendidero\Germanized\Shipments\Interfaces\ShippingProviderAuto' ) ) { + return false; + } + + $classname = $shipping_provider->get_label_classname( $label_type ); + + /** + * Filter that allows adjusting the default DHL label classname. + * + * @param string $classname The classname to be used. + * @param integer $label_id The label id. + * @param string $label_type The label type. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + $classname = apply_filters( 'woocommerce_gzd_shipment_label_class', $classname, $label_id, $label_type, $shipping_provider ); + + if ( ! class_exists( $classname ) ) { + return false; + } + + try { + return new $classname( $label_id ); + } catch ( Exception $e ) { + wc_caught_exception( $e, __FUNCTION__, array( $label_id, $shipping_provider_name, $label_type ) ); + return false; + } + } + + public static function get_label_id( $label ) { + if ( is_numeric( $label ) ) { + return $label; + } elseif ( $label instanceof Label ) { + return $label->get_id(); + } elseif ( ! empty( $label->label_id ) ) { + return $label->label_id; + } else { + return false; + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Labels/Label.php b/packages/woocommerce-germanized-shipments/src/Labels/Label.php new file mode 100644 index 000000000..96ab1dfb6 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Labels/Label.php @@ -0,0 +1,861 @@ + null, + 'shipment_id' => 0, + 'product_id' => '', + 'parent_id' => 0, + 'number' => '', + 'shipping_provider' => '', + 'weight' => '', + 'net_weight' => '', + 'length' => '', + 'width' => '', + 'height' => '', + 'path' => '', + 'created_via' => '', + 'services' => array(), + ); + + public function __construct( $data = 0 ) { + parent::__construct( $data ); + + if ( $data instanceof ShipmentLabel ) { + $this->set_id( absint( $data->get_id() ) ); + } elseif ( is_numeric( $data ) ) { + $this->set_id( $data ); + } + + $this->data_store = WC_Data_Store::load( $this->data_store_name ); + + // If we have an ID, load the user from the DB. + if ( $this->get_id() ) { + try { + $this->data_store->read( $this ); + } catch ( Exception $e ) { + $this->set_id( 0 ); + $this->set_object_read( true ); + } + } else { + $this->set_object_read( true ); + } + } + + public function get_type() { + return 'simple'; + } + + /** + * Merge changes with data and clear. + * Overrides WC_Data::apply_changes. + * array_replace_recursive does not work well for license because it merges domains registered instead + * of replacing them. + * + * @since 3.2.0 + */ + public function apply_changes() { + if ( function_exists( 'array_replace' ) ) { + $this->data = array_replace( $this->data, $this->changes ); // phpcs:ignore PHPCompatibility.FunctionUse.NewFunctions.array_replaceFound + } else { // PHP 5.2 compatibility. + foreach ( $this->changes as $key => $change ) { + $this->data[ $key ] = $change; + } + } + $this->changes = array(); + } + + /** + * Prefix for action and filter hooks on data. + * + * @since 3.0.0 + * @return string + */ + protected function get_general_hook_prefix() { + $prefix = 'simple' === $this->get_type() ? '' : $this->get_type() . '_'; + + return "woocommerce_gzd_shipment_{$prefix}label_"; + } + + /** + * Prefix for action and filter hooks on data. + * + * @since 3.0.0 + * @return string + */ + protected function get_hook_prefix() { + return $this->get_general_hook_prefix() . 'get_'; + } + + /** + * Return the date this license was created. + * + * @since 3.0.0 + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return WC_DateTime|null object if the date is set or null if there is no date. + */ + public function get_date_created( $context = 'view' ) { + return $this->get_prop( 'date_created', $context ); + } + + public function get_shipment_id( $context = 'view' ) { + return $this->get_prop( 'shipment_id', $context ); + } + + public function get_shipping_provider( $context = 'view' ) { + return $this->get_prop( 'shipping_provider', $context ); + } + + public function get_shipping_provider_instance() { + $provider = $this->get_shipping_provider(); + + if ( ! empty( $provider ) ) { + return wc_gzd_get_shipping_provider( $provider ); + } + + return false; + } + + public function get_parent_id( $context = 'view' ) { + return $this->get_prop( 'parent_id', $context ); + } + + public function get_created_via( $context = 'view' ) { + return $this->get_prop( 'created_via', $context ); + } + + public function get_product_id( $context = 'view' ) { + return $this->get_prop( 'product_id', $context ); + } + + public function get_number( $context = 'view' ) { + return $this->get_prop( 'number', $context ); + } + + public function has_number() { + $number = $this->get_number(); + + return empty( $number ) ? false : true; + } + + /** + * Returns the weight in kg + * + * @param string $context + * + * @return string + */ + public function get_weight( $context = 'view' ) { + return $this->get_prop( 'weight', $context ); + } + + public function get_net_weight( $context = 'view' ) { + $weight = $this->get_prop( 'net_weight', $context ); + + if ( 'view' === $context && '' === $weight ) { + $weight = $this->get_weight( $context ); + } + + return $weight; + } + + /** + * Returns the length in cm + * + * @param string $context + * + * @return string + */ + public function get_length( $context = 'view' ) { + return $this->get_prop( 'length', $context ); + } + + /** + * Returns the width in cm + * + * @param string $context + * + * @return string + */ + public function get_width( $context = 'view' ) { + return $this->get_prop( 'width', $context ); + } + + /** + * Returns the height in cm + * + * @param string $context + * + * @return string + */ + public function get_height( $context = 'view' ) { + return $this->get_prop( 'height', $context ); + } + + public function get_dimensions( $context = 'view' ) { + return array( + 'length' => $this->get_length( $context ), + 'width' => $this->get_width( $context ), + 'height' => $this->get_height( $context ), + ); + } + + public function has_dimensions() { + $width = $this->get_width(); + $length = $this->get_length(); + $height = $this->get_height(); + + return ( ! empty( $width ) && ! empty( $length ) && ! empty( $height ) ); + } + + public function get_path( $context = 'view', $file_path = '' ) { + return $this->get_prop( 'path', $context ); + } + + public function get_services( $context = 'view' ) { + return $this->get_prop( 'services', $context ); + } + + public function has_service( $service ) { + return ( in_array( $service, $this->get_services(), true ) ); + } + + public function get_shipment() { + if ( is_null( $this->shipment ) ) { + $this->shipment = ( $this->get_shipment_id() > 0 ? wc_gzd_get_shipment( $this->get_shipment_id() ) : false ); + } + + return $this->shipment; + } + + /* + |-------------------------------------------------------------------------- + | Setters + |-------------------------------------------------------------------------- + */ + + /** + * Set the date this license was last updated. + * + * @since 1.0.0 + * @param string|integer|null $date UTC timestamp, or ISO 8601 DateTime. If the DateTime string has no timezone or offset, WordPress site timezone will be assumed. Null if their is no date. + */ + public function set_date_created( $date = null ) { + $this->set_date_prop( 'date_created', $date ); + } + + public function set_number( $number ) { + $this->set_prop( 'number', $number ); + } + + public function set_product_id( $number ) { + $this->set_prop( 'product_id', $number ); + } + + public function set_shipping_provider( $slug ) { + $this->set_prop( 'shipping_provider', $slug ); + } + + public function set_parent_id( $id ) { + $this->set_prop( 'parent_id', absint( $id ) ); + } + + public function set_created_via( $created_via ) { + $this->set_prop( 'created_via', $created_via ); + } + + public function set_weight( $weight ) { + $this->set_prop( 'weight', '' !== $weight ? wc_format_decimal( $weight ) : '' ); + } + + public function set_net_weight( $weight ) { + $this->set_prop( 'net_weight', '' !== $weight ? wc_format_decimal( $weight ) : '' ); + } + + public function set_width( $width ) { + $this->set_prop( 'width', '' !== $width ? wc_format_decimal( $width ) : '' ); + } + + public function set_length( $length ) { + $this->set_prop( 'length', '' !== $length ? wc_format_decimal( $length ) : '' ); + } + + public function set_height( $height ) { + $this->set_prop( 'height', '' !== $height ? wc_format_decimal( $height ) : '' ); + } + + public function set_path( $path, $file_type = '' ) { + $this->set_prop( 'path', $path ); + } + + public function set_services( $services ) { + $this->set_prop( 'services', empty( $services ) ? array() : (array) $services ); + } + + /** + * Returns linked children labels. + * + * @return ShipmentLabel[] + */ + public function get_children() { + return wc_gzd_get_shipment_labels( array( 'parent_id' => $this->get_id() ) ); + } + + public function has_children() { + $children = $this->get_children(); + + return count( $children ) > 0 ? true : false; + } + + public function add_service( $service ) { + $services = (array) $this->get_services(); + $available_services = array(); + + if ( $provider = $this->get_shipping_provider_instance() ) { + if ( $shipment = $this->get_shipment() ) { + $available_services = $provider->get_available_label_services( $shipment ); + } + } + + if ( ! in_array( $service, $services, true ) && in_array( $service, $available_services, true ) ) { + $services[] = $service; + $this->set_services( $services ); + + return true; + } + + return false; + } + + public function remove_service( $service ) { + $services = (array) $this->get_services(); + + if ( in_array( $service, $services, true ) ) { + $services = array_diff( $services, array( $service ) ); + + $this->set_services( $services ); + return true; + } + + return false; + } + + public function supports_additional_file_type( $file_type ) { + return in_array( $file_type, $this->get_additional_file_types(), true ); + } + + public function get_additional_file_types() { + return array(); + } + + public function get_file( $file_type = '' ) { + if ( ! $path = $this->get_path( 'view', $file_type ) ) { + return false; + } + + return $this->get_file_by_path( $path ); + } + + protected function get_new_filename( $file_type = '' ) { + $file_parts = array( + $this->get_shipping_provider(), + ); + + if ( ! empty( $file_type ) ) { + $file_parts[] = $file_type; + } + + if ( 'simple' !== $this->get_type() ) { + $file_parts[] = $this->get_type(); + } + + $file_parts[] = $this->get_shipment_id(); + + $filename_default = implode( '-', $file_parts ); + $filename_default = $filename_default . '.pdf'; + $filename = apply_filters( "{$this->get_hook_prefix()}filename", $filename_default, $this, $file_type ); + + return sanitize_file_name( $filename ); + } + + public function get_filename( $file_type = '' ) { + if ( ! $path = $this->get_path( 'view', $file_type ) ) { + return $this->get_new_filename( $file_type ); + } + + return basename( $path ); + } + + protected function get_file_by_path( $file ) { + // If the file is relative, prepend upload dir. + if ( $file && 0 !== strpos( $file, '/' ) && ( ( $uploads = Package::get_upload_dir() ) && false === $uploads['error'] ) ) { + $file = $uploads['basedir'] . "/$file"; + + return $file; + } else { + return false; + } + } + + public function set_shipment_id( $shipment_id ) { + // Reset order object + $this->shipment = null; + + $this->set_prop( 'shipment_id', absint( $shipment_id ) ); + } + + /** + * @param Shipment $shipment + */ + public function set_shipment( &$shipment ) { + $this->shipment = $shipment; + + $this->set_prop( 'shipment_id', absint( $shipment->get_id() ) ); + } + + public function get_download_url( $args = array() ) { + $base_url = is_admin() ? admin_url() : trailingslashit( home_url() ); + $download_url = add_query_arg( + array( + 'action' => 'wc-gzd-download-shipment-label', + 'shipment_id' => $this->get_shipment_id(), + ), + wp_nonce_url( $base_url, 'download-shipment-label' ) + ); + + foreach ( $args as $arg => $val ) { + if ( is_bool( $val ) ) { + $args[ $arg ] = wc_bool_to_string( $val ); + } + } + + $download_url = add_query_arg( $args, $download_url ); + + /** + * Filter for shipping providers to adjust the label download URL. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. `$provider` is related to the current shipping provider + * for the shipment (slug). + * + * Example hook name: `woocommerce_gzd_return_shipment_get_dhl_label_download_url` + * + * @param string $url The download URL. + * @param Label $label The current shipment instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return esc_url_raw( apply_filters( "{$this->get_hook_prefix()}download_url", $download_url, $this ) ); + } + + /** + * @return ShipmentError|true + */ + public function fetch() { + $result = new ShipmentError( 'label-fetch-error', _x( 'This label misses the API implementation', 'shipments', 'woocommerce-germanized' ) ); + + return $result; + } + + /** + * @param $stream + * @param string $file_type + * + * @return false|string + */ + public function upload_label_file( $stream, $file_type = '' ) { + try { + Package::set_upload_dir_filter(); + $filename = $this->get_filename( $file_type ); + + $GLOBALS['gzd_shipments_unique_filename'] = $filename; + add_filter( 'wp_unique_filename', '_wc_gzd_shipments_keep_force_filename', 10, 1 ); + + $tmp = wp_upload_bits( $this->get_filename( $file_type ), null, $stream ); + + unset( $GLOBALS['gzd_shipments_unique_filename'] ); + remove_filter( 'wp_unique_filename', '_wc_gzd_shipments_keep_force_filename', 10 ); + + Package::unset_upload_dir_filter(); + + if ( isset( $tmp['file'] ) ) { + $path = $tmp['file']; + $path = Package::get_relative_upload_dir( $path ); + + $this->set_path( $path, $file_type ); + + return $path; + } else { + throw new Exception( _x( 'Error while uploading label.', 'shipments', 'woocommerce-germanized' ) ); + } + } catch ( Exception $e ) { + return false; + } + } + + /** + * @param $url + * @param string $file_type + * + * @return false|string + */ + public function download_label_file( $url, $file_type = '' ) { + $timeout_seconds = 5; + + try { + if ( ! function_exists( 'download_url' ) ) { + include_once ABSPATH . 'wp-admin/includes/file.php'; + } + + if ( ! function_exists( 'download_url' ) ) { + throw new \Exception( _x( 'Error while downloading the PDF file.', 'shipments', 'woocommerce-germanized' ) ); + } + + // Download file to temp dir. + $temp_file = download_url( $url, $timeout_seconds ); + + if ( is_wp_error( $temp_file ) ) { + throw new \Exception( _x( 'Error while downloading the PDF file.', 'shipments', 'woocommerce-germanized' ) ); + } + + $file = array( + 'name' => $this->get_filename( $file_type ), + 'type' => 'application/pdf', + 'tmp_name' => $temp_file, + 'error' => 0, + 'size' => filesize( $temp_file ), + ); + + $overrides = array( + 'test_type' => false, + 'test_form' => false, + 'test_size' => true, + ); + + // Move the temporary file into the uploads directory. + Package::set_upload_dir_filter(); + $results = wp_handle_sideload( $file, $overrides ); + Package::unset_upload_dir_filter(); + + if ( empty( $results['error'] ) ) { + $path = Package::get_relative_upload_dir( $results['file'] ); + + $this->set_path( $path, $file_type ); + + return $path; + } else { + throw new \Exception( _x( 'Error while downloading the PDF file.', 'shipments', 'woocommerce-germanized' ) ); + } + } catch ( \Exception $e ) { + return false; + } + } + + public function is_trackable() { + return true; + } + + public function supports_third_party_email_notification() { + $supports_email_notification = false; + + if ( ( $shipment = $this->get_shipment() ) && ( $order = $shipment->get_order() ) ) { + $supports_email_notification = wc_gzd_order_supports_parcel_delivery_reminder( $order ); + } + + return apply_filters( "{$this->get_general_hook_prefix()}supports_third_party_email_notification", $supports_email_notification, $this ); + } + + /** + * Gets a prop for a getter method. + * + * @since 3.0.0 + * @param string $prop Name of prop to get. + * @param string $address billing or shipping. + * @param string $context What the value is for. Valid values are view and edit. + * @return mixed + */ + protected function get_address_prop( $prop, $address = 'sender_address', $context = 'view' ) { + $value = null; + + if ( isset( $this->changes[ $address ][ $prop ] ) || isset( $this->data[ $address ][ $prop ] ) ) { + $value = isset( $this->changes[ $address ][ $prop ] ) ? $this->changes[ $address ][ $prop ] : $this->data[ $address ][ $prop ]; + + if ( 'view' === $context ) { + /** + * Filter to adjust a specific address property for a DHL label. + * + * The dynamic portion of the hook name, `$this->get_hook_prefix()` constructs an individual + * hook name which uses `woocommerce_gzd_dhl_label_get_` as a prefix. Additionally + * `$address` contains the current address type e.g. sender_address and `$prop` contains the actual + * property e.g. street. + * + * Example hook name: `woocommerce_gzd_dhl_return_label_get_sender_address_street` + * + * @param string $value The address property value. + * @param \Vendidero\Germanized\DHL\Label\Label $label The label object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + $value = apply_filters( "{$this->get_hook_prefix()}{$address}_{$prop}", $value, $this ); + } + } + + return $value; + } + + protected function round_customs_item_weight( $value, $precision = 0 ) { + return \Automattic\WooCommerce\Utilities\NumberUtil::round( $value, $precision, 2 ); + } + + protected function get_per_item_weights( $total_weight, $item_weights, $shipment_items ) { + $item_total_weight = array_sum( $item_weights ); + $item_count = count( $item_weights ); + + /** + * Discrepancies detected between item weights an total shipment weight. + * Try to distribute the mismatch between items. + */ + if ( $item_total_weight != $total_weight ) { // phpcs:ignore WordPress.PHP.StrictComparisons.LooseComparison + $diff = $total_weight - $item_total_weight; + $diff_abs = abs( $diff ); + + if ( $diff_abs > 0 ) { + $per_item_diff = $diff / $item_count; + // Round down to int + $per_item_diff_rounded = $this->round_customs_item_weight( $per_item_diff ); + $diff_applied = 0; + + if ( abs( $per_item_diff_rounded ) > 0 ) { + foreach ( $item_weights as $key => $weight ) { + $shipment_item = $shipment_items[ $key ]; + $item_min_weight = 1 * $shipment_item->get_quantity(); + + $item_weight_before = $item_weights[ $key ]; + $new_item_weight = $item_weights[ $key ] += $per_item_diff_rounded; + $item_diff_applied = $per_item_diff_rounded; + + /** + * In case the diff is negative make sure we are not + * subtracting more than available as min weight per item. + */ + if ( $new_item_weight <= $item_min_weight ) { + $new_item_weight = $item_min_weight; + $item_diff_applied = $item_min_weight - $item_weight_before; + } + + $item_weights[ $key ] = $new_item_weight; + $diff_applied += $item_diff_applied; + } + } + + // Check rounding diff and apply the diff to one item + $diff_left = $diff - $diff_applied; + + if ( abs( $diff_left ) > 0 ) { + foreach ( $item_weights as $key => $weight ) { + $shipment_item = $shipment_items[ $key ]; + $item_min_weight = 1 * $shipment_item->get_quantity(); + + if ( $diff_left > 0 ) { + /** + * Add the diff left to the first item and stop. + */ + $item_weights[ $key ] += $diff_left; + break; + } else { + /** + * Remove the diff left from the first item with a weight greater than 0.01 to prevent 0 weights. + */ + if ( $weight > $item_min_weight ) { + $item_weights[ $key ] += $diff_left; + break; + } + } + } + } + } + } + + return $item_weights; + } + + public function get_customs_data( $max_desc_length = 255 ) { + if ( ! $shipment = $this->get_shipment() ) { + return false; + } + + $customs_items = array(); + $item_description = ''; + $total_weight = $this->round_customs_item_weight( wc_add_number_precision( $this->get_net_weight() ) ); + $total_gross_weight = $this->round_customs_item_weight( wc_add_number_precision( $this->get_weight() ) ); + $item_weights = array(); + $shipment_items = $shipment->get_items(); + $order = $shipment->get_order(); + + foreach ( $shipment_items as $key => $item ) { + $per_item_weight = wc_format_decimal( floatval( wc_get_weight( $item->get_weight(), 'kg', $shipment->get_weight_unit() ) ), 2 ); + $per_item_weight = wc_add_number_precision( $per_item_weight ); + $per_item_weight = $per_item_weight * $item->get_quantity(); + $per_item_min_weight = 1 * $item->get_quantity(); + + /** + * Set min weight to 0.01 to prevent missing weight error messages + * for really small product weights. + */ + if ( $per_item_weight < $per_item_min_weight ) { + $per_item_weight = $per_item_min_weight; + } + + $item_weights[ $key ] = $per_item_weight; + } + + $item_weights = $this->get_per_item_weights( $total_weight, $item_weights, $shipment_items ); + $item_gross_weights = $this->get_per_item_weights( $total_gross_weight, $item_weights, $shipment_items ); + $total_weight = 0; + $total_gross_weight = 0; + $total_value = 0; + $use_subtotal = false; + + if ( $order && apply_filters( 'woocommerce_gzd_shipments_order_has_voucher', false, $order ) ) { + $use_subtotal = true; + } + + $use_subtotal = apply_filters( 'woocommerce_gzd_shipments_customs_use_subtotal', $use_subtotal, $this ); + + foreach ( $shipment->get_items() as $key => $item ) { + $item_description .= ! empty( $item_description ) ? ', ' : ''; + $item_description .= $item->get_name(); + + // Use total before discounts for customs + $product_total = floatval( (float) ( $use_subtotal ? $item->get_subtotal() : $item->get_total() ) / $item->get_quantity() ); + $dhl_product = false; + $product = $item->get_product(); + + if ( $product ) { + $shipment_product = wc_gzd_shipments_get_product( $product ); + } + + if ( $product_total < 0.01 ) { + // Use the order item data as fallback + if ( ( $order_item = $item->get_order_item() ) && $order ) { + $order_item_total = $use_subtotal ? $order->get_line_subtotal( $order_item, true, false ) : $order->get_line_total( $order_item, true, false ); + $product_total = floatval( (float) $order_item_total / $item->get_quantity() ); + } + } + + $category = $shipment_product ? $shipment_product->get_main_category() : $item->get_name(); + + if ( empty( $category ) ) { + $category = $item->get_name(); + } + + $product_value = $product_total < 0.01 ? wc_format_decimal( apply_filters( "{$this->get_general_hook_prefix()}customs_item_min_price", 0.01, $item, $this, $shipment ), 2 ) : wc_format_decimal( $product_total, 2 ); + + $customs_items[ $key ] = apply_filters( + "{$this->get_general_hook_prefix()}customs_item", + array( + 'description' => apply_filters( "{$this->get_general_hook_prefix()}item_description", wc_clean( mb_substr( $item->get_name(), 0, $max_desc_length ) ), $item, $this, $shipment ), + 'category' => apply_filters( "{$this->get_general_hook_prefix()}item_category", $category, $item, $this, $shipment ), + 'origin_code' => ( $shipment_product && $shipment_product->get_manufacture_country() ) ? $shipment_product->get_manufacture_country() : Package::get_base_country(), + 'tariff_number' => $shipment_product ? $shipment_product->get_hs_code() : '', + 'quantity' => intval( $item->get_quantity() ), + 'weight_in_kg' => wc_remove_number_precision( $item_weights[ $key ] ), + 'single_weight_in_kg' => $this->round_customs_item_weight( wc_remove_number_precision( $item_weights[ $key ] / $item->get_quantity() ), 2 ), + 'weight_in_kg_raw' => $item_weights[ $key ], + 'gross_weight_in_kg' => wc_remove_number_precision( $item_gross_weights[ $key ] ), + 'single_gross_weight_in_kg' => $this->round_customs_item_weight( wc_remove_number_precision( $item_gross_weights[ $key ] / $item->get_quantity() ), 2 ), + 'gross_weight_in_kg_raw' => $item_gross_weights[ $key ], + 'single_value' => $product_value, + 'value' => wc_format_decimal( $product_value * $item->get_quantity(), 2 ), + ), + $item, + $shipment, + $this + ); + + $total_weight += (float) $customs_items[ $key ]['weight_in_kg']; + $total_gross_weight += (float) $customs_items[ $key ]['gross_weight_in_kg']; + $total_value += (float) $customs_items[ $key ]['value']; + } + + $item_description = mb_substr( $item_description, 0, $max_desc_length ); + + $customs_data = apply_filters( + "{$this->get_general_hook_prefix()}customs_data", + array( + 'shipment_id' => $shipment->get_id(), + 'additional_fee' => wc_format_decimal( $shipment->get_additional_total(), 2 ), + 'export_type_description' => $item_description, + 'place_of_commital' => $shipment->get_sender_city(), + // e.g. EORI number + 'sender_customs_ref_number' => $shipment->get_sender_customs_reference_number(), + 'receiver_customs_ref_number' => $shipment->get_customs_reference_number(), + // Customs UK VAT ID (HMRC) for totals <= 135 GBP + 'sender_customs_uk_vat_id' => $shipment->get_sender_customs_uk_vat_id(), + 'items' => $customs_items, + 'item_total_weight_in_kg' => $total_weight, + 'item_total_gross_weight_in_kg' => $total_gross_weight, + 'item_total_value' => $total_value, + 'currency' => $order ? $order->get_currency() : get_woocommerce_currency(), + 'invoice_number' => '', + ), + $this, + $shipment + ); + + return $customs_data; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Labels/Query.php b/packages/woocommerce-germanized-shipments/src/Labels/Query.php new file mode 100644 index 000000000..8a2d8c8eb --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Labels/Query.php @@ -0,0 +1,490 @@ + 10, + 'shipment_id' => '', + 'product_id' => '', + 'shipping_provider' => '', + 'parent_id' => '', + 'type' => wc_gzd_get_shipment_label_types(), + 'number' => '', + 'order' => 'DESC', + 'orderby' => 'date_created', + 'return' => 'objects', + 'page' => 1, + 'offset' => '', + 'paginate' => false, + 'search' => '', + 'search_columns' => array(), + ); + } + + /** + * Get labels matching the current query vars. + * + * @return Label[] objects + * + * @throws \Exception When WC_Data_Store validation fails. + */ + public function get_labels() { + /** + * Filter to adjust query paramaters for a DHL label query. + * + * @param array $query_vars The query arguments. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + $args = apply_filters( 'woocommerce_gzd_shipment_label_query_args', $this->get_query_vars() ); + $args = WC_Data_Store::load( 'shipment-label' )->get_query_args( $args ); + + $this->query( $args ); + + /** + * Filter to adjust query result data for a DHL label query. + * + * @param Label[] $results The results. + * @param array $args The query arguments. + * + * @since 3.0.0 + * @package Vendidero/Germanized/DHL + */ + return apply_filters( 'woocommerce_gzd_shipment_label_query', $this->results, $args ); + } + + public function get_total() { + return $this->total_labels; + } + + /** + * Query shipments. + * + * @param array $query_args + */ + protected function query( $query_args ) { + global $wpdb; + + $this->args = $query_args; + $this->parse_query(); + $this->prepare_query(); + + $qv =& $this->args; + + $this->results = null; + + if ( null === $this->results ) { + $this->request = "SELECT $this->query_fields $this->query_from $this->query_where $this->query_orderby $this->query_limit"; + + if ( is_array( $qv['fields'] ) || 'objects' === $qv['fields'] ) { + $this->results = $wpdb->get_results( $this->request ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } else { + $this->results = $wpdb->get_col( $this->request ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } + + if ( isset( $qv['count_total'] ) && $qv['count_total'] ) { + $found_labels_query = 'SELECT FOUND_ROWS()'; + $this->total_labels = (int) $wpdb->get_var( $found_labels_query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } + } + + if ( ! $this->results ) { + return; + } + + if ( 'objects' === $qv['fields'] ) { + foreach ( $this->results as $key => $label ) { + $this->results[ $key ] = wc_gzd_get_shipment_label( $label ); + } + } + } + + /** + * Parse the query before preparing it. + */ + protected function parse_query() { + + if ( isset( $this->args['shipment_id'] ) ) { + $this->args['shipment_id'] = absint( $this->args['shipment_id'] ); + } + + if ( isset( $this->args['shipping_provider'] ) ) { + $this->args['shipping_provider'] = sanitize_key( $this->args['shipping_provider'] ); + } + + if ( isset( $this->args['product_id'] ) ) { + $this->args['product_id'] = wc_clean( $this->args['product_id'] ); + } + + if ( isset( $this->args['parent_id'] ) && ! empty( $this->args['parent_id'] ) ) { + $this->args['parent_id'] = absint( $this->args['parent_id'] ); + } + + if ( isset( $this->args['number'] ) ) { + $this->args['number'] = sanitize_key( $this->args['number'] ); + } + + if ( isset( $this->args['type'] ) ) { + $this->args['type'] = (array) $this->args['type']; + $this->args['type'] = array_map( 'wc_clean', $this->args['type'] ); + } + + if ( isset( $this->args['search'] ) ) { + $this->args['search'] = wc_clean( $this->args['search'] ); + + if ( ! isset( $this->args['search_columns'] ) ) { + $this->args['search_columns'] = array(); + } + } + } + + /** + * Prepare the query for DB usage. + */ + protected function prepare_query() { + global $wpdb; + + if ( is_array( $this->args['fields'] ) ) { + $this->args['fields'] = array_unique( $this->args['fields'] ); + + $this->query_fields = array(); + + foreach ( $this->args['fields'] as $field ) { + $field = 'ID' === $field ? 'label_id' : sanitize_key( $field ); + $this->query_fields[] = "$wpdb->gzd_shipment_labels.$field"; + } + + $this->query_fields = implode( ',', $this->query_fields ); + + } elseif ( 'objects' === $this->args['fields'] ) { + $this->query_fields = "$wpdb->gzd_shipment_labels.*"; + } else { + $this->query_fields = "$wpdb->gzd_shipment_labels.label_id"; + } + + if ( isset( $this->args['count_total'] ) && $this->args['count_total'] ) { + $this->query_fields = 'SQL_CALC_FOUND_ROWS ' . $this->query_fields; + } + + $this->query_from = "FROM $wpdb->gzd_shipment_labels"; + $this->query_where = 'WHERE 1=1'; + + // order id + if ( isset( $this->args['shipment_id'] ) ) { + $this->query_where .= $wpdb->prepare( ' AND label_shipment_id = %d', $this->args['shipment_id'] ); + } + + // parent id + if ( isset( $this->args['parent_id'] ) ) { + $this->query_where .= $wpdb->prepare( ' AND label_parent_id = %d', $this->args['parent_id'] ); + } + + // Number + if ( isset( $this->args['number'] ) ) { + $this->query_where .= $wpdb->prepare( " AND label_number IN ('%s')", $this->args['number'] ); // phpcs:ignore WordPress.DB.PreparedSQLPlaceholders.QuotedSimplePlaceholder + } + + // Shipping provider + if ( isset( $this->args['shipping_provider'] ) ) { + $this->query_where .= $wpdb->prepare( " AND label_shipping_provider IN ('%s')", $this->args['shipping_provider'] ); // phpcs:ignore WordPress.DB.PreparedSQLPlaceholders.QuotedSimplePlaceholder + } + + // Shipping provider + if ( isset( $this->args['product_id'] ) ) { + $this->query_where .= $wpdb->prepare( " AND label_product_id IN ('%s')", $this->args['product_id'] ); // phpcs:ignore WordPress.DB.PreparedSQLPlaceholders.QuotedSimplePlaceholder + } + + // type + if ( isset( $this->args['type'] ) ) { + $types = $this->args['type']; + $p_types = array(); + + foreach ( $types as $type ) { + $p_types[] = $wpdb->prepare( 'label_type = %s', $type ); + } + + $where_type = implode( ' OR ', $p_types ); + + if ( ! empty( $where_type ) ) { + $this->query_where .= " AND ($where_type)"; + } + } + + // Search + $search = ''; + + if ( isset( $this->args['search'] ) ) { + $search = trim( $this->args['search'] ); + } + + if ( $search ) { + + $leading_wild = ( ltrim( $search, '*' ) !== $search ); + $trailing_wild = ( rtrim( $search, '*' ) !== $search ); + + if ( $leading_wild && $trailing_wild ) { + $wild = 'both'; + } elseif ( $leading_wild ) { + $wild = 'leading'; + } elseif ( $trailing_wild ) { + $wild = 'trailing'; + } else { + $wild = false; + } + if ( $wild ) { + $search = trim( $search, '*' ); + } + + $search_columns = array(); + + if ( $this->args['search_columns'] ) { + $search_columns = array_intersect( $this->args['search_columns'], array( 'label_id', 'label_path', 'label_number', 'label_shipment_id', 'label_product_id' ) ); + } + + if ( ! $search_columns ) { + if ( is_numeric( $search ) ) { + $search_columns = array( 'label_id', 'label_shipment_id' ); + } else { + $search_columns = array( 'label_id', 'label_path', 'label_number', 'label_shipment_id', 'label_product_id' ); + } + } + + /** + * Filters the columns to search in a DHL LabelQuery search. + * + * The default columns depend on the search term, and include 'label_id', + * 'label_shipment_id', 'label_path' and 'label_number'. + * + * @since 3.0.0 + * + * @param string[] $search_columns Array of column names to be searched. + * @param string $search Text being searched. + * @param LabelQuery $this The current LabelQuery instance. + * + * @package Vendidero/Germanized/DHL + */ + $search_columns = apply_filters( 'woocommerce_gzd_shipment_label_search_columns', $search_columns, $search, $this ); + + $this->query_where .= $this->get_search_sql( $search, $search_columns, $wild ); + } + + // Parse and sanitize 'include', for use by 'orderby' as well as 'include' below. + if ( ! empty( $this->args['include'] ) ) { + $include = wp_parse_id_list( $this->args['include'] ); + } else { + $include = false; + } + + // Meta query. + $this->meta_query = new WP_Meta_Query(); + $this->meta_query->parse_query_vars( $this->args ); + + if ( ! empty( $this->meta_query->queries ) ) { + $clauses = $this->meta_query->get_sql( 'gzd_shipment_label', $wpdb->gzd_shipment_labels, 'label_id', $this ); + $this->query_from .= $clauses['join']; + $this->query_where .= $clauses['where']; + + if ( $this->meta_query->has_or_relation() ) { + $this->query_fields = 'DISTINCT ' . $this->query_fields; + } + } + + // sorting + $this->args['order'] = isset( $this->args['order'] ) ? strtoupper( $this->args['order'] ) : ''; + $order = $this->parse_order( $this->args['order'] ); + + if ( empty( $this->args['orderby'] ) ) { + // Default order is by 'user_login'. + $ordersby = array( 'date_created' => $order ); + } elseif ( is_array( $this->args['orderby'] ) ) { + $ordersby = $this->args['orderby']; + } else { + // 'orderby' values may be a comma- or space-separated list. + $ordersby = preg_split( '/[,\s]+/', $this->args['orderby'] ); + } + + $orderby_array = array(); + + foreach ( $ordersby as $_key => $_value ) { + if ( ! $_value ) { + continue; + } + + if ( is_int( $_key ) ) { + // Integer key means this is a flat array of 'orderby' fields. + $_orderby = $_value; + $_order = $order; + } else { + // Non-integer key means this the key is the field and the value is ASC/DESC. + $_orderby = $_key; + $_order = $_value; + } + + $parsed = $this->parse_orderby( $_orderby ); + + if ( ! $parsed ) { + continue; + } + + $orderby_array[] = $parsed . ' ' . $this->parse_order( $_order ); + } + + // If no valid clauses were found, order by user_login. + if ( empty( $orderby_array ) ) { + $orderby_array[] = "label_id $order"; + } + + $this->query_orderby = 'ORDER BY ' . implode( ', ', $orderby_array ); + + // limit + if ( isset( $this->args['posts_per_page'] ) && $this->args['posts_per_page'] > 0 ) { + if ( isset( $this->args['offset'] ) ) { + $this->query_limit = $wpdb->prepare( 'LIMIT %d, %d', $this->args['offset'], $this->args['posts_per_page'] ); + } else { + $this->query_limit = $wpdb->prepare( 'LIMIT %d, %d', $this->args['posts_per_page'] * ( $this->args['page'] - 1 ), $this->args['posts_per_page'] ); + } + } + + if ( ! empty( $include ) ) { + // Sanitized earlier. + $ids = implode( ',', $include ); + $this->query_where .= " AND $wpdb->gzd_shipment_labels.label_id IN ($ids)"; + } elseif ( ! empty( $this->args['exclude'] ) ) { + $ids = implode( ',', wp_parse_id_list( $this->args['exclude'] ) ); + $this->query_where .= " AND $wpdb->gzd_shipment_labels.label_id NOT IN ($ids)"; + } + + // Date queries are allowed for the user_registered field. + if ( ! empty( $this->args['date_query'] ) && is_array( $this->args['date_query'] ) ) { + $date_query = new WP_Date_Query( $this->args['date_query'], 'label_date_created' ); + $this->query_where .= $date_query->get_sql(); + } + } + + /** + * Used internally to generate an SQL string for searching across multiple columns + * + * @since 3.0.6 + * + * @global \wpdb $wpdb WordPress database abstraction object. + * + * @param string $string + * @param array $cols + * @param bool $wild Whether to allow wildcard searches. Default is false for Network Admin, true for single site. + * Single site allows leading and trailing wildcards, Network Admin only trailing. + * @return string + */ + protected function get_search_sql( $string, $cols, $wild = false ) { + global $wpdb; + + $searches = array(); + $leading_wild = ( 'leading' === $wild || 'both' === $wild ) ? '%' : ''; + $trailing_wild = ( 'trailing' === $wild || 'both' === $wild ) ? '%' : ''; + $like = $leading_wild . $wpdb->esc_like( $string ) . $trailing_wild; + + foreach ( $cols as $col ) { + if ( 'ID' === $col ) { + $searches[] = $wpdb->prepare( "$col = %s", $string ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared + } else { + $searches[] = $wpdb->prepare( "$col LIKE %s", $like ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared + } + } + + return ' AND (' . implode( ' OR ', $searches ) . ')'; + } + + /** + * Parse orderby statement. + * + * @param string $orderby + * @return string + */ + protected function parse_orderby( $orderby ) { + global $wpdb; + + $meta_query_clauses = $this->meta_query->get_clauses(); + + $_orderby = ''; + + if ( in_array( $orderby, array( 'number', 'shipment_id', 'date_created' ), true ) ) { + $_orderby = 'label_' . $orderby; + } elseif ( 'ID' === $orderby || 'id' === $orderby ) { + $_orderby = 'label_id'; + } elseif ( 'meta_value' === $orderby || $this->get( 'meta_key' ) === $orderby ) { + $_orderby = "$wpdb->gzd_shipment_labelmeta.meta_value"; + } elseif ( 'meta_value_num' === $orderby ) { + $_orderby = "$wpdb->gzd_shipment_labelmeta.meta_value+0"; + } elseif ( 'include' === $orderby && ! empty( $this->args['include'] ) ) { + $include = wp_parse_id_list( $this->args['include'] ); + $include_sql = implode( ',', $include ); + $_orderby = "FIELD( $wpdb->gzd_shipment_labels.label_id, $include_sql )"; + } elseif ( isset( $meta_query_clauses[ $orderby ] ) ) { + $meta_clause = $meta_query_clauses[ $orderby ]; + $_orderby = sprintf( 'CAST(%s.meta_value AS %s)', esc_sql( $meta_clause['alias'] ), esc_sql( $meta_clause['cast'] ) ); + } + + return $_orderby; + } + + /** + * Parse order statement. + * + * @param string $order + * @return string + */ + protected function parse_order( $order ) { + if ( ! is_string( $order ) || empty( $order ) ) { + return 'DESC'; + } + + if ( 'ASC' === strtoupper( $order ) ) { + return 'ASC'; + } else { + return 'DESC'; + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Labels/ReturnLabel.php b/packages/woocommerce-germanized-shipments/src/Labels/ReturnLabel.php new file mode 100644 index 000000000..53b3494a7 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Labels/ReturnLabel.php @@ -0,0 +1,119 @@ + array(), + ); + + public function get_type() { + return 'return'; + } + + public function get_sender_address( $context = 'view' ) { + return $this->get_prop( 'sender_address', $context ); + } + + /** + * Gets a prop for a getter method. + * + * @since 3.0.0 + * @param string $prop Name of prop to get. + * @param string $address billing or shipping. + * @param string $context What the value is for. Valid values are view and edit. + * @return mixed + */ + protected function get_sender_address_prop( $prop, $context = 'view' ) { + $value = $this->get_address_prop( $prop, 'sender_address', $context ); + + return $value; + } + + public function get_sender_address_2( $context = 'view' ) { + return $this->get_sender_address_prop( 'address_2', $context ); + } + + public function get_sender_address_addition() { + $addition = $this->get_sender_address_2(); + $street_addition = $this->get_sender_street_addition(); + + if ( ! empty( $street_addition ) ) { + $addition = $street_addition . ( ! empty( $addition ) ? ' ' . $addition : '' ); + } + + return trim( $addition ); + } + + public function get_sender_street( $context = 'view' ) { + return $this->get_sender_address_prop( 'street', $context ); + } + + public function get_sender_street_number( $context = 'view' ) { + return $this->get_sender_address_prop( 'street_number', $context ); + } + + public function get_sender_street_addition( $context = 'view' ) { + return $this->get_sender_address_prop( 'street_addition', $context ); + } + + public function get_sender_company( $context = 'view' ) { + return $this->get_sender_address_prop( 'company', $context ); + } + + public function get_sender_name( $context = 'view' ) { + return $this->get_sender_address_prop( 'name', $context ); + } + + public function get_sender_formatted_full_name() { + return sprintf( _x( '%1$s', 'shipments full name', 'woocommerce-germanized' ), $this->get_sender_name() ); // phpcs:ignore WordPress.WP.I18n.NoEmptyStrings + } + + public function get_sender_postcode( $context = 'view' ) { + return $this->get_sender_address_prop( 'postcode', $context ); + } + + public function get_sender_city( $context = 'view' ) { + return $this->get_sender_address_prop( 'city', $context ); + } + + public function get_sender_state( $context = 'view' ) { + return $this->get_sender_address_prop( 'state', $context ); + } + + public function get_sender_country( $context = 'view' ) { + return $this->get_sender_address_prop( 'country', $context ); + } + + public function get_sender_phone( $context = 'view' ) { + return $this->get_sender_address_prop( 'phone', $context ); + } + + public function get_sender_email( $context = 'view' ) { + return $this->get_sender_address_prop( 'email', $context ); + } + + public function set_sender_address( $value ) { + $this->set_prop( 'sender_address', empty( $value ) ? array() : (array) $value ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Order.php b/packages/woocommerce-germanized-shipments/src/Order.php new file mode 100644 index 000000000..5ab084465 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Order.php @@ -0,0 +1,1009 @@ +order = $order; + } + + /** + * Returns the Woo WC_Order original object + * + * @return object|WC_Order + */ + public function get_order() { + return $this->order; + } + + /** + * @return WC_DateTime|null + */ + public function get_date_shipped() { + $date_shipped = $this->get_order()->get_meta( '_date_shipped', true ); + + if ( $date_shipped ) { + try { + $date_shipped = new WC_DateTime( "@{$date_shipped}" ); + + // Set local timezone or offset. + if ( get_option( 'timezone_string' ) ) { + $date_shipped->setTimezone( new DateTimeZone( wc_timezone_string() ) ); + } else { + $date_shipped->set_utc_offset( wc_timezone_offset() ); + } + } catch ( Exception $e ) { + $date_shipped = null; + } + } else { + $date_shipped = null; + } + + return $date_shipped; + } + + public function is_shipped() { + $shipping_status = $this->get_shipping_status(); + + return apply_filters( 'woocommerce_gzd_shipment_order_shipping_status', ( in_array( $shipping_status, array( 'shipped', 'delivered' ), true ) || ( 'partially-delivered' === $shipping_status && ! $this->needs_shipping( array( 'sent_only' => true ) ) ) ), $this ); + } + + public function get_shipping_status() { + $status = 'not-shipped'; + $shipments = $this->get_simple_shipments(); + $all_shipments_delivered = false; + $all_shipments_shipped = false; + + if ( ! empty( $shipments ) ) { + $all_shipments_delivered = true; + $all_shipments_shipped = true; + + foreach ( $shipments as $shipment ) { + if ( ! $shipment->has_status( 'delivered' ) ) { + $all_shipments_delivered = false; + } else { + $status = 'partially-delivered'; + } + + if ( ! $shipment->is_shipped() ) { + $all_shipments_shipped = false; + } elseif ( 'partially-delivered' !== $status ) { + $status = 'partially-shipped'; + } + } + } + + $needs_shipping = $this->needs_shipping( array( 'sent_only' => true ) ); + + if ( $all_shipments_delivered && ! $needs_shipping ) { + $status = 'delivered'; + } elseif ( 'partially-delivered' !== $status && ( $all_shipments_shipped && ! $needs_shipping ) ) { + $status = 'shipped'; + } elseif ( ! in_array( $status, array( 'partially-shipped', 'partially-delivered' ), true ) && ! $needs_shipping ) { + $status = 'no-shipping-needed'; + } + + return apply_filters( 'woocommerce_gzd_shipment_order_shipping_status', $status, $this ); + } + + public function has_shipped_shipments() { + $shipments = $this->get_simple_shipments(); + + foreach ( $shipments as $shipment ) { + if ( $shipment->is_shipped() ) { + return true; + } + } + + return false; + } + + public function get_return_status() { + $status = 'open'; + $shipments = $this->get_return_shipments(); + + if ( ! empty( $shipments ) ) { + foreach ( $shipments as $shipment ) { + if ( $shipment->has_status( 'delivered' ) ) { + $status = 'partially-returned'; + break; + } + } + } + + if ( ! $this->needs_return( array( 'delivered_only' => true ) ) && $this->has_shipped_shipments() ) { + $status = 'returned'; + } + + return $status; + } + + public function get_default_return_shipping_provider() { + $default_provider_instance = wc_gzd_get_order_shipping_provider( $this->get_order() ); + $default_provider = $default_provider_instance ? $default_provider_instance->get_name() : ''; + $shipments = $this->get_simple_shipments(); + + foreach ( $shipments as $shipment ) { + if ( $shipment->is_shipped() ) { + $default_provider = $shipment->get_shipping_provider(); + } + } + + return apply_filters( 'woocommerce_gzd_shipment_order_return_default_shipping_provider', $default_provider, $this ); + } + + public function validate_shipments( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'save' => true, + ) + ); + + foreach ( $this->get_simple_shipments() as $shipment ) { + if ( $shipment->is_editable() ) { + + // Make sure we are working based on the current instance. + $shipment->set_order_shipment( $this ); + $shipment->sync(); + + $this->validate_shipment_item_quantities( $shipment->get_id() ); + } + } + + if ( $args['save'] ) { + $this->save(); + } + } + + /** + * @param Shipment $shipment + * + * @return float + */ + public function calculate_shipment_additional_total( $shipment ) { + $fees_total = 0.0; + + foreach ( $this->get_order()->get_fees() as $item ) { + $fees_total += ( (float) $item->get_total() + (float) $item->get_total_tax() ); + } + + $additional_total = $fees_total + (float) $this->get_order()->get_shipping_total() + (float) $this->get_order()->get_shipping_tax(); + + foreach ( $this->get_simple_shipments() as $simple_shipment ) { + if ( $shipment->get_id() === $simple_shipment->get_id() ) { + continue; + } + + $additional_total -= (float) $simple_shipment->get_additional_total(); + } + + $additional_total = wc_format_decimal( $additional_total, '' ); + + if ( (float) $additional_total < 0.0 ) { + $additional_total = 0.0; + } + + return $additional_total; + } + + public function validate_shipment_item_quantities( $shipment_id = false ) { + $shipment = $shipment_id ? $this->get_shipment( $shipment_id ) : false; + $shipments = ( $shipment_id && $shipment ) ? array( $shipment ) : $this->get_simple_shipments(); + $order_items = $this->get_shippable_items(); + + foreach ( $shipments as $shipment ) { + + if ( ! is_a( $shipment, 'Vendidero\Germanized\Shipments\Shipment' ) ) { + continue; + } + + // Do only check draft shipments + if ( $shipment->is_editable() ) { + foreach ( $shipment->get_items() as $item ) { + + // Order item does not exist + if ( ! isset( $order_items[ $item->get_order_item_id() ] ) ) { + + /** + * Filter to decide whether to keep non-existing OrderItems within + * the Shipment while validating or not. + * + * @param boolean $keep Whether to keep non-existing OrderItems or not. + * @param ShipmentItem $item The shipment item object. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + if ( ! apply_filters( 'woocommerce_gzd_shipment_order_keep_non_order_item', false, $item, $shipment ) ) { + $shipment->remove_item( $item->get_id() ); + } + + continue; + } + + $order_item = $order_items[ $item->get_order_item_id() ]; + $quantity = $this->get_item_quantity_left_for_shipping( + $order_item, + array( + 'shipment_id' => $shipment->get_id(), + 'exclude_current_shipment' => true, + ) + ); + + if ( $quantity <= 0 ) { + $shipment->remove_item( $item->get_id() ); + } else { + $new_quantity = absint( $item->get_quantity() ); + + if ( $item->get_quantity() > $quantity ) { + $new_quantity = $quantity; + } + + $item->sync( array( 'quantity' => $new_quantity ) ); + } + } + + if ( empty( $shipment->get_items() ) ) { + $this->remove_shipment( $shipment->get_id() ); + } + } + } + } + + /** + * @return Shipment[] Shipments + */ + public function get_shipments() { + + if ( is_null( $this->shipments ) ) { + + $this->shipments = wc_gzd_get_shipments( + array( + 'order_id' => $this->get_order()->get_id(), + 'limit' => -1, + 'orderby' => 'date_created', + 'type' => array( 'simple', 'return' ), + 'order' => 'ASC', + ) + ); + } + + $shipments = (array) $this->shipments; + + return $shipments; + } + + /** + * @return SimpleShipment[] + */ + public function get_simple_shipments( $shipped_only = false ) { + $simple = array(); + + foreach ( $this->get_shipments() as $shipment ) { + if ( 'simple' === $shipment->get_type() ) { + if ( $shipped_only && ! $shipment->is_shipped() ) { + continue; + } + + $simple[] = $shipment; + } + } + + return $simple; + } + + /** + * @return ReturnShipment[] + */ + public function get_return_shipments() { + $returns = array(); + + foreach ( $this->get_shipments() as $shipment ) { + if ( 'return' === $shipment->get_type() ) { + $returns[] = $shipment; + } + } + + return $returns; + } + + public function add_shipment( &$shipment ) { + $shipments = $this->get_shipments(); + + $this->shipments[] = $shipment; + } + + public function remove_shipment( $shipment_id ) { + $shipments = $this->get_shipments(); + + foreach ( $this->shipments as $key => $shipment ) { + if ( $shipment->get_id() === (int) $shipment_id ) { + $this->shipments_to_delete[] = $shipment; + + unset( $this->shipments[ $key ] ); + break; + } + } + } + + /** + * @param $shipment_id + * + * @return bool|SimpleShipment|ReturnShipment + */ + public function get_shipment( $shipment_id ) { + $shipments = $this->get_shipments(); + + foreach ( $shipments as $shipment ) { + + if ( $shipment->get_id() === (int) $shipment_id ) { + return $shipment; + } + } + + return false; + } + + /** + * @param WC_Order_Item $order_item + */ + public function get_item_quantity_left_for_shipping( $order_item, $args = array() ) { + $quantity_left = 0; + $args = wp_parse_args( + $args, + array( + 'sent_only' => false, + 'shipment_id' => 0, + 'exclude_current_shipment' => false, + ) + ); + + if ( is_numeric( $order_item ) ) { + $order_item = $this->get_order()->get_item( $order_item ); + } + + if ( $order_item ) { + $quantity_left = $this->get_shippable_item_quantity( $order_item ); + + foreach ( $this->get_shipments() as $shipment ) { + + if ( $args['sent_only'] && ! $shipment->is_shipped() ) { + continue; + } + + if ( $args['exclude_current_shipment'] && $args['shipment_id'] > 0 && ( $shipment->get_id() === $args['shipment_id'] ) ) { + continue; + } + + if ( $item = $shipment->get_item_by_order_item_id( $order_item->get_id() ) ) { + if ( 'return' === $shipment->get_type() ) { + if ( $shipment->is_shipped() ) { + $quantity_left += absint( $item->get_quantity() ); + } + } else { + $quantity_left -= absint( $item->get_quantity() ); + } + } + } + } + + if ( $quantity_left < 0 ) { + $quantity_left = 0; + } + + /** + * Filter to adjust the quantity left for shipment of a specific order item. + * + * @param integer $quantity_left The quantity left for shipment. + * @param WC_Order_Item $order_item The order item object. + * @param Order $this The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_item_quantity_left_for_shipping', $quantity_left, $order_item, $this ); + } + + public function get_item_quantity_sent_by_order_item_id( $order_item_id ) { + $shipments = $this->get_simple_shipments(); + $quantity = 0; + + foreach ( $shipments as $shipment ) { + + if ( ! $shipment->is_shipped() ) { + continue; + } + + if ( $item = $shipment->get_item_by_order_item_id( $order_item_id ) ) { + $quantity += absint( $item->get_quantity() ); + } + } + + return $quantity; + } + + /** + * @param ShipmentItem $item + */ + public function get_item_quantity_left_for_returning( $order_item_id, $args = array() ) { + $quantity_left = 0; + $args = wp_parse_args( + $args, + array( + 'delivered_only' => false, + 'shipment_id' => 0, + 'exclude_current_shipment' => false, + ) + ); + + $quantity_left = $this->get_item_quantity_sent_by_order_item_id( $order_item_id ); + + foreach ( $this->get_return_shipments() as $shipment ) { + + if ( $args['delivered_only'] && ! $shipment->has_status( 'delivered' ) ) { + continue; + } + + if ( $args['exclude_current_shipment'] && $args['shipment_id'] > 0 && ( $shipment->get_id() === $args['shipment_id'] ) ) { + continue; + } + + if ( $shipment_item = $shipment->get_item_by_order_item_id( $order_item_id ) ) { + $quantity_left -= absint( $shipment_item->get_quantity() ); + } + } + + if ( $quantity_left < 0 ) { + $quantity_left = 0; + } + + /** + * Filter to adjust the quantity left for returning of a specific order item. + * + * @param integer $quantity_left The quantity left for shipment. + * @param integer $order_item_id The order item id. + * @param Order $this The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_item_quantity_left_for_returning', $quantity_left, $order_item_id, $this ); + } + + /** + * @param false $group_by_shipping_class + * + * @return OrderItem[] + */ + public function get_items_to_pack_left_for_shipping( $group_by_shipping_class = false ) { + $items = $this->get_available_items_for_shipment(); + $items_to_be_packed = array(); + + foreach ( $items as $order_item_id => $item ) { + if ( ! $order_item = $this->get_order()->get_item( $order_item_id ) ) { + continue; + } + + $shipping_class = ''; + + if ( $group_by_shipping_class ) { + if ( $product = $order_item->get_product() ) { + $shipping_class = $product->get_shipping_class(); + } + } + + for ( $i = 0; $i < $item['max_quantity']; $i++ ) { + try { + $box_item = new Packing\OrderItem( $order_item ); + + if ( ! isset( $items_to_be_packed[ $shipping_class ] ) ) { + $items_to_be_packed[ $shipping_class ] = array(); + } + + $items_to_be_packed[ $shipping_class ][] = $box_item; + } catch ( \Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } + } + } + + return apply_filters( 'woocommerce_gzd_shipment_order_items_to_pack_left_for_shipping', $items_to_be_packed, $group_by_shipping_class ); + } + + /** + * @param bool|Shipment $shipment + * @return array + */ + public function get_available_items_for_shipment( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'disable_duplicates' => false, + 'shipment_id' => 0, + 'sent_only' => false, + 'exclude_current_shipment' => false, + ) + ); + + $items = array(); + $shipment = $args['shipment_id'] ? $this->get_shipment( $args['shipment_id'] ) : false; + + foreach ( $this->get_shippable_items() as $item ) { + $quantity_left = $this->get_item_quantity_left_for_shipping( $item, $args ); + + if ( $shipment ) { + if ( $args['disable_duplicates'] && $shipment->get_item_by_order_item_id( $item->get_id() ) ) { + continue; + } + } + + if ( $quantity_left > 0 ) { + $sku = ''; + + if ( is_callable( array( $item, 'get_product' ) ) ) { + if ( $product = $item->get_product() ) { + $sku = $product->get_sku(); + } + } + + $items[ $item->get_id() ] = array( + 'name' => $item->get_name() . ( ! empty( $sku ) ? ' (' . esc_html( $sku ) . ')' : '' ), + 'max_quantity' => $quantity_left, + ); + } + } + + return $items; + } + + /** + * Returns the first found matching shipment item for a certain order item id. + * + * @param $order_item_id + * + * @return bool|ShipmentItem + */ + public function get_simple_shipment_item( $order_item_id ) { + foreach ( $this->get_simple_shipments() as $shipment ) { + + if ( $item = $shipment->get_item_by_order_item_id( $order_item_id ) ) { + return $item; + } + } + + return false; + } + + /** + * @return array + */ + public function get_available_items_for_return( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'disable_duplicates' => false, + 'shipment_id' => 0, + 'delivered_only' => false, + 'exclude_current_shipment' => false, + ) + ); + + $items = array(); + $shipment = $args['shipment_id'] ? $this->get_shipment( $args['shipment_id'] ) : false; + + foreach ( $this->get_returnable_items() as $item ) { + $quantity_left = $this->get_item_quantity_left_for_returning( $item->get_order_item_id(), $args ); + + if ( $shipment ) { + if ( $args['disable_duplicates'] && $shipment->get_item_by_order_item_id( $item->get_order_item_id() ) ) { + continue; + } + } + + if ( $quantity_left > 0 ) { + $sku = $item->get_sku(); + + $items[ $item->get_order_item_id() ] = array( + 'name' => $item->get_name() . ( ! empty( $sku ) ? ' (' . esc_html( $sku ) . ')' : '' ), + 'max_quantity' => $quantity_left, + ); + } + } + + return $items; + } + + public function item_needs_shipping( $order_item, $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'sent_only' => false, + ) + ); + + $needs_shipping = false; + + if ( $this->get_item_quantity_left_for_shipping( $order_item, $args ) > 0 ) { + $needs_shipping = true; + } + + /** + * Filter to decide whether an order item needs shipping or not. + * + * @param boolean $needs_shipping Whether the item needs shipping or not. + * @param WC_Order_Item $item The order item object. + * @param array $args Additional arguments to be considered. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_item_needs_shipping', $needs_shipping, $order_item, $args, $this ); + } + + /** + * Checks whether an item needs return or not by checking the quantity left for return. + * + * @param ShipmentItem $item + * @param array $args + * + * @return mixed|void + */ + public function item_needs_return( $item, $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'delivered_only' => false, + ) + ); + + $needs_return = false; + + if ( $this->get_item_quantity_left_for_returning( $item->get_order_item_id(), $args ) > 0 ) { + $needs_return = true; + } + + /** + * Filter to decide whether a shipment item needs return or not. + * + * @param boolean $needs_return Whether the item needs return or not. + * @param ShipmentItem $item The order item object. + * @param array $args Additional arguments to be considered. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_item_needs_return', $needs_return, $item, $args, $this ); + } + + /** + * Returns the return request key added to allow a guest customer to add + * a new return request to a certain order. + * + * @return mixed + */ + public function get_order_return_request_key() { + return $this->get_order()->get_meta( '_return_request_key' ); + } + + /** + * Removes the return request key from the order. Saves the order. + */ + public function delete_order_return_request_key() { + $this->get_order()->delete_meta_data( '_return_request_key' ); + $this->get_order()->save(); + } + + /** + * Returns items that are ready for shipping (defaults to non-virtual line items). + * + * @return WC_Order_Item[] Shippable items. + */ + public function get_shippable_items() { + $items = $this->get_order()->get_items( 'line_item' ); + + foreach ( $items as $key => $item ) { + $product = is_callable( array( $item, 'get_product' ) ) ? $item->get_product() : false; + + if ( $product ) { + if ( $product->is_virtual() || $this->get_shippable_item_quantity( $item ) <= 0 ) { + unset( $items[ $key ] ); + } + } + } + + $items = array_filter( $items ); + + /** + * Filter to adjust shippable order items for a specific order. + * By default excludes virtual items. + * + * @param WC_Order_Item[] $items Array containing shippable order items. + * @param WC_Order $order The order object. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_shippable_items', $items, $this->get_order(), $this ); + } + + /** + * Returns items that are ready for return. By default only shipped (or delivered) items are returnable. + * + * @return ShipmentItem[] Shippable items. + */ + public function get_returnable_items() { + $items = array(); + + foreach ( $this->get_simple_shipments() as $shipment ) { + + if ( ! $shipment->is_shipped() ) { + continue; + } + + foreach ( $shipment->get_items() as $item ) { + + if ( ! isset( $items[ $item->get_order_item_id() ] ) ) { + $new_item = clone $item; + $items[ $item->get_order_item_id() ] = $new_item; + } else { + $new_quantity = absint( $items[ $item->get_order_item_id() ]->get_quantity() ) + absint( $item->get_quantity() ); + $items[ $item->get_order_item_id() ]->set_quantity( $new_quantity ); + } + } + } + + /** + * Filter to adjust returnable items for a specific order. + * + * @param ShipmentItem[] $items Array containing shippable order items. + * @param WC_Order $order The order object. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_returnable_items', $items, $this->get_order(), $this ); + } + + public function get_shippable_item_quantity( $order_item ) { + $refunded_qty = absint( $this->get_order()->get_qty_refunded_for_item( $order_item->get_id() ) ); + + // Make sure we are safe to substract quantity for logical purposes + if ( $refunded_qty < 0 ) { + $refunded_qty *= -1; + } + + $quantity_left = absint( $order_item->get_quantity() ) - $refunded_qty; + + /** + * Filter that allows adjusting the quantity left for shipping or a specific order item. + * + * @param integer $quantity_left The quantity left for shipping. + * @param WC_Order_Item $item The order item object. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_item_shippable_quantity', $quantity_left, $order_item, $this ); + } + + /** + * Returns the total number of shippable items. + * + * @return mixed|void + */ + public function get_shippable_item_count() { + $count = 0; + + foreach ( $this->get_shippable_items() as $item ) { + $count += $this->get_shippable_item_quantity( $item ); + } + + /** + * Filters the total number of shippable items available in an order. + * + * @param integer $count The total number of items. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_shippable_item_count', $count, $this ); + } + + /** + * Returns the number of total returnable items. + * + * @return mixed|void + */ + public function get_returnable_item_count() { + $count = 0; + + foreach ( $this->get_returnable_items() as $item ) { + $count += absint( $item->get_quantity() ); + } + + /** + * Filters the total number of returnable items available in an order. + * + * @param integer $count The total number of items. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_returnable_item_count', $count, $this ); + } + + protected function has_local_pickup() { + $shipping_methods = $this->get_order()->get_shipping_methods(); + $has_pickup = false; + + /** + * Filters which shipping methods are considered local pickup method + * which by default do not require shipment. + * + * @param string[] $pickup_methods Array of local pickup shipping method ids. + * + * @since 3.1.6 + * @package Vendidero/Germanized/Shipments + */ + $pickup_methods = apply_filters( 'woocommerce_gzd_shipment_local_pickup_shipping_methods', array( 'local_pickup' ) ); + + foreach ( $shipping_methods as $shipping_method ) { + if ( in_array( $shipping_method->get_method_id(), $pickup_methods, true ) ) { + $has_pickup = true; + break; + } + } + + return $has_pickup; + } + + /** + * Checks whether the order needs shipping or not by checking quantity + * for every line item. + * + * @param bool $sent_only Whether to only include shipments treated as sent or not. + * + * @return bool Whether the order needs shipping or not. + */ + public function needs_shipping( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'sent_only' => false, + ) + ); + + $order_items = $this->get_shippable_items(); + $needs_shipping = false; + $has_pickup = $this->has_local_pickup(); + + if ( ! $has_pickup ) { + foreach ( $order_items as $order_item ) { + if ( $this->item_needs_shipping( $order_item, $args ) ) { + $needs_shipping = true; + break; + } + } + } + + /** + * Filter to decide whether an order needs shipping or not. + * + * @param boolean $needs_shipping Whether the order needs shipping or not. + * @param WC_Order $order The order object. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_needs_shipping', $needs_shipping, $this->get_order(), $this ); + } + + /** + * Checks whether the order needs return or not by checking quantity + * for every line item. + * + * @return bool Whether the order needs shipping or not. + */ + public function needs_return( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'delivered_only' => false, + ) + ); + + $items = $this->get_returnable_items(); + $needs_return = false; + + foreach ( $items as $item ) { + + if ( $this->item_needs_return( $item, $args ) ) { + $needs_return = true; + break; + } + } + + /** + * Filter to decide whether an order needs return or not. + * + * @param boolean $needs_return Whether the order needs return or not. + * @param WC_Order $order The order object. + * @param Order $order The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_order_needs_return', $needs_return, $this->get_order(), $this ); + } + + public function save() { + if ( ! empty( $this->shipments_to_delete ) ) { + + foreach ( $this->shipments_to_delete as $shipment ) { + $shipment->delete( true ); + } + } + + foreach ( $this->shipments as $shipment ) { + $shipment->save(); + } + } + + /** + * Call child methods if the method does not exist. + * + * @param $method + * @param $args + * + * @return bool|mixed + */ + public function __call( $method, $args ) { + + if ( method_exists( $this->order, $method ) ) { + return call_user_func_array( array( $this->order, $method ), $args ); + } + + return false; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/PDFMerger.php b/packages/woocommerce-germanized-shipments/src/PDFMerger.php new file mode 100644 index 000000000..41131dfc2 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/PDFMerger.php @@ -0,0 +1,142 @@ +_pdf = new Fpdi(); + } + + /** + * Add file to this pdf + * + * @param string $filename Filename of the source file + * @param mixed $pages Range of files (if not set, all pages where imported) + */ + public function add( $filename, $pages = array(), $width = 210 ) { + if ( file_exists( $filename ) ) { + $page_count = $this->_pdf->setSourceFile( $filename ); + + for ( $i = 1; $i <= $page_count; $i ++ ) { + if ( $this->_isPageInRange( $i, $pages ) ) { + $this->_addPage( $i, $width ); + } + } + } + + return $this; + } + + /** + * Output merged pdf + * + * @param string $type + */ + public function output( $filename, $type = 'I' ) { + return $this->_pdf->Output( $type, $filename ); + } + + /** + * Force download merged pdf as file + * + * @param $filename + * + * @return string + */ + public function download( $filename ) { + return $this->output( $filename, 'D' ); + } + + /** + * Save merged pdf + * + * @param $filename + * + * @return string + */ + public function save( $filename ) { + return $this->output( $filename, 'F' ); + } + + /** + * Add single page + * + * @param $page_number + * + * @throws PdfReaderException + */ + private function _addPage( $page_number, $width = 210 ) { // phpcs:ignore PSR2.Methods.MethodDeclaration.Underscore,WordPress.NamingConventions.ValidFunctionName.MethodNameInvalid + $page_id = $this->_pdf->importPage( $page_number ); + $size = $this->_pdf->getTemplateSize( $page_id ); + + $orientation = isset( $size['orientation'] ) ? $size['orientation'] : ''; + + $this->_pdf->addPage( $orientation, $size ); + + if ( ! isset( $size['width'] ) || empty( $size['width'] ) ) { + $this->_pdf->useImportedPage( $page_id, 0, 0, $width, null, true ); + } else { + $this->_pdf->useImportedPage( $page_id ); + } + } + + + /** + * Check if a specific page should be merged. + * If pages are empty, all pages will be merged + * + * @return bool + */ + private function _isPageInRange( $page_number, $pages = array() ) { // phpcs:ignore PSR2.Methods.MethodDeclaration.Underscore,WordPress.NamingConventions.ValidFunctionName.MethodNameInvalid + if ( empty( $pages ) ) { + return true; + } + + foreach ( $pages as $range ) { + if ( in_array( $page_number, $this->_getRange( $range ), true ) ) { + return true; + } + } + + return false; + } + + + /** + * Get range by given value + * + * @param mixed $value + * + * @return array + */ + private function _getRange( $value = null ) { // phpcs:ignore PSR2.Methods.MethodDeclaration.Underscore,WordPress.NamingConventions.ValidFunctionName.MethodNameInvalid + $value = preg_replace( '/[^0-9\-.]/is', '', $value ); + + if ( '' === $value ) { + return false; + } + + $value = explode( '-', $value ); + + if ( 1 === count( $value ) ) { + return $value; + } + + return range( $value[0] > $value[1] ? $value[1] : $value[0], $value[0] > $value[1] ? $value[0] : $value[1] ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/PDFSplitter.php b/packages/woocommerce-germanized-shipments/src/PDFSplitter.php new file mode 100644 index 000000000..ef518b1dd --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/PDFSplitter.php @@ -0,0 +1,177 @@ +_pdf = new Fpdi(); + + try { + + if ( $stream ) { + $file = StreamReader::createByString( $file ); + $this->filename = $filename; + } else { + $this->filename = basename( $this->file ); + } + + $this->file = $file; + $this->pagecount = $this->_pdf->setSourceFile( $file ); // How many pages? + + } catch ( PdfParserException $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } catch ( Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } + } + + public function get_page_count() { + return $this->pagecount; + } + + public function split() { + $new_files = array(); + + try { + // Split each page into a new PDF + for ( $i = 1; $i <= $this->pagecount; $i++ ) { + + $new_pdf = new Fpdi(); + $new_pdf->AddPage(); + $new_pdf->setSourceFile( $this->file ); + $new_pdf->useTemplate( $new_pdf->importPage( $i ), 0, 0, 210, null, true ); + + $new_files[] = $new_pdf->Output( 'S', $this->filename ); + } + } catch ( PdfParserException $e ) { + return false; + } + + return $new_files; + } + + /** + * Add file to this pdf + * + * @param string $filename Filename of the source file + * @param mixed $pages Range of files (if not set, all pages where imported) + */ + public function add( $filename, $pages = array() ) { + if ( file_exists( $filename ) ) { + $page_count = $this->_pdf->setSourceFile( $filename ); + for ( $i = 1; $i <= $page_count; $i++ ) { + if ( $this->_isPageInRange( $i, $pages ) ) { + $this->_addPage( $i ); + } + } + } + return $this; + } + + /** + * Output merged pdf + * + * @param string $type + */ + public function output( $filename, $type = 'I' ) { + return $this->_pdf->Output( $type, $filename ); + } + + /** + * Force download merged pdf as file + * + * @param $filename + * @return string + */ + public function download( $filename ) { + return $this->output( $filename, 'D' ); + } + + /** + * Save merged pdf + * + * @param $filename + * @return string + */ + public function save( $filename ) { + return $this->output( $filename, 'F' ); + } + + /** + * Add single page + * + * @param $page_number + * @throws PdfReaderException + */ + private function _addPage( $page_number ) { // phpcs:ignore PSR2.Methods.MethodDeclaration.Underscore,WordPress.NamingConventions.ValidFunctionName.MethodNameInvalid + $page_id = $this->_pdf->importPage( $page_number ); + $this->_pdf->addPage(); + $this->_pdf->useImportedPage( $page_id ); + } + + + /** + * Check if a specific page should be merged. + * If pages are empty, all pages will be merged + * + * @return bool + */ + private function _isPageInRange( $page_number, $pages = array() ) { // phpcs:ignore PSR2.Methods.MethodDeclaration.Underscore,WordPress.NamingConventions.ValidFunctionName.MethodNameInvalid + if ( empty( $pages ) ) { + return true; + } + + foreach ( $pages as $range ) { + if ( in_array( $page_number, $this->_getRange( $range ), true ) ) { + return true; + } + } + + return false; + } + + + /** + * Get range by given value + * + * @param mixed $value + * @return array + */ + private function _getRange( $value = null ) { // phpcs:ignore PSR2.Methods.MethodDeclaration.Underscore,WordPress.NamingConventions.ValidFunctionName.MethodNameInvalid + $value = preg_replace( '/[^0-9\-.]/is', '', $value ); + + if ( '' === $value ) { + return false; + } + + $value = explode( '-', $value ); + + if ( 1 === count( $value ) ) { + return $value; + } + + return range( $value[0] > $value[1] ? $value[1] : $value[0], $value[0] > $value[1] ? $value[0] : $value[1] ); + } + +} diff --git a/packages/woocommerce-germanized-shipments/src/Package.php b/packages/woocommerce-germanized-shipments/src/Package.php new file mode 100644 index 000000000..e60f7ecc5 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Package.php @@ -0,0 +1,713 @@ +query_vars['add-return-shipment'] ) ) { + $callback = 'woocommerce_gzd_shipments_template_add_return_shipment'; + $order_id = absint( $wp->query_vars['add-return-shipment'] ); + } + + if ( $callback && $order_id && ( $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) && ! empty( $key ) ) { + + // Order return key is invalid. + if ( ! wc_gzd_customer_can_add_return_shipment( $order_id ) ) { + throw new Exception( _x( 'Sorry, this order is invalid and cannot be returned.', 'shipments', 'woocommerce-germanized' ) ); + } else { + call_user_func_array( $callback, array( 'order_id' => $order_id ) ); + $template = self::get_path() . '/templates/global/empty.php'; + } + } + } catch ( Exception $e ) { + wc_add_notice( $e->getMessage(), 'error' ); + } + } + + return $template; + } + + public static function register_shortcodes() { + add_shortcode( 'gzd_return_request_form', array( __CLASS__, 'return_request_form' ) ); + } + + public static function return_request_form( $args = array() ) { + $defaults = array( + 'message' => '', + 'hidden' => false, + ); + + $args = wp_parse_args( $args, $defaults ); + $notices = function_exists( 'wc_print_notices' ) ? wc_print_notices( true ) : ''; + $html = ''; + + // Output notices in case notices have not been outputted yet. + if ( ! empty( $notices ) ) { + $html .= '
' . $notices . '
'; + } + + $html .= wc_get_template_html( 'global/form-return-request.php', $args ); + + return $html; + } + + public static function load_wpml_compatibility( $compatibility ) { + WPMLHelper::init( $compatibility ); + } + + public static function set_method_filters( $methods ) { + foreach ( $methods as $method => $class ) { + add_filter( 'woocommerce_shipping_instance_form_fields_' . $method, array( __CLASS__, 'add_method_settings' ), 10, 1 ); + /** + * Use this filter as a backup to support plugins like Flexible Shipping which may override methods + */ + add_filter( 'woocommerce_settings_api_form_fields_' . $method, array( __CLASS__, 'add_method_settings' ), 10, 1 ); + add_filter( 'woocommerce_shipping_' . $method . '_instance_settings_values', array( __CLASS__, 'filter_method_settings' ), 10, 2 ); + } + + return $methods; + } + + /** + * Indicates whether the BoxPack library for improved packing calculation is supported + * + * @return bool + */ + public static function is_packing_supported() { + return version_compare( phpversion(), '7.1', '>=' ) && apply_filters( 'woocommerce_gzd_enable_rucksack_packaging', true ); + } + + public static function get_method_settings() { + if ( is_null( self::$method_settings ) ) { + self::$method_settings = Method::get_admin_settings(); + } + + return self::$method_settings; + } + + public static function filter_method_settings( $p_settings, $method ) { + $shipping_provider_settings = self::get_method_settings(); + $shipping_provider = isset( $p_settings['shipping_provider'] ) ? $p_settings['shipping_provider'] : ''; + $shipping_method = wc_gzd_get_shipping_provider_method( $method ); + + /** + * Make sure the (maybe) new selected provider is used on updating the settings. + */ + $shipping_method->set_provider( $shipping_provider ); + + foreach ( $p_settings as $setting => $value ) { + if ( array_key_exists( $setting, $shipping_provider_settings ) ) { + // Check if setting does neither belong to global setting nor shipping provider prefix + if ( 'shipping_provider' !== $setting && ! $shipping_method->setting_belongs_to_provider( $setting ) ) { + unset( $p_settings[ $setting ] ); + } elseif ( $shipping_method->get_fallback_setting_value( $setting ) === $value ) { + unset( $p_settings[ $setting ] ); + } elseif ( '' === $value ) { + unset( $p_settings[ $setting ] ); + } + } + } + + /** + * Filter that returns shipping method settings cleaned from global shipping provider method settings. + * This filter might be useful to remove some default setting values from + * shipping provider method settings e.g. DHL settings. + * + * @param array $p_settings The settings + * @param WC_Shipping_Method $method The shipping method instance + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipping_provider_method_clean_settings', $p_settings, $method ); + } + + public static function add_method_settings( $p_settings ) { + $wc = WC(); + + /** + * Prevent undefined index notices during REST API calls. + * + * @see WC_REST_Shipping_Zone_Methods_V2_Controller::get_settings() + */ + if ( is_callable( array( $wc, 'is_rest_api_request' ) ) && $wc->is_rest_api_request() ) { + return $p_settings; + } + + $shipping_provider_settings = self::get_method_settings(); + + return array_merge( $p_settings, $shipping_provider_settings ); + } + + public static function load_shipping_methods( $package ) { + $shipping = WC_Shipping::instance(); + + foreach ( $shipping->shipping_methods as $key => $method ) { + $shipping_provider_method = new Method( $method ); + } + } + + public static function is_hpos_enabled() { + if ( ! is_callable( array( '\Automattic\WooCommerce\Utilities\OrderUtil', 'custom_orders_table_usage_is_enabled' ) ) ) { + return false; + } + + return \Automattic\WooCommerce\Utilities\OrderUtil::custom_orders_table_usage_is_enabled(); + } + + public static function inject_endpoints() { + if ( function_exists( 'WC' ) && WC()->query ) { + foreach ( self::get_endpoints() as $endpoint ) { + if ( ! array_key_exists( $endpoint, WC()->query->query_vars ) ) { + $option_name = str_replace( '-', '_', $endpoint ); + WC()->query->query_vars[ $endpoint ] = get_option( "woocommerce_gzd_shipments_{$option_name}_endpoint", $endpoint ); + } + } + } + } + + public static function get_base_country() { + $default_country = wc_get_base_location()['country']; + $shipment_country = wc_format_country_state_string( self::get_setting( 'shipper_address_country' ) )['country']; + + if ( empty( $shipment_country ) ) { + $shipment_country = $default_country; + } + + return apply_filters( 'woocommerce_gzd_shipment_base_country', $shipment_country ); + } + + public static function get_base_postcode() { + $default_postcode = WC()->countries->get_base_postcode(); + $shipment_postcode = self::get_setting( 'shipper_address_postcode' ); + + if ( empty( $shipment_postcode ) ) { + $shipment_postcode = $default_postcode; + } + + return apply_filters( 'woocommerce_gzd_shipment_base_postcode', $shipment_postcode ); + } + + public static function base_country_belongs_to_eu_customs_area() { + return self::country_belongs_to_eu_customs_area( self::get_base_country(), self::get_base_postcode() ); + } + + public static function country_belongs_to_eu_customs_area( $country, $postcode = '' ) { + $country = wc_strtoupper( $country ); + $eu_countries = WC()->countries->get_european_union_countries(); + $belongs = false; + $postcode = wc_normalize_postcode( $postcode ); + $postcode_wildcards = wc_get_wildcard_postcodes( $postcode, $country ); + + if ( in_array( $country, $eu_countries, true ) ) { + $belongs = true; + } + + if ( $belongs ) { + $exemptions = array( + 'DE' => array( + '27498', // Helgoland + '78266', // Büsingen am Hochrhein + ), + 'ES' => array( + '35*', // Canary Islands + '38*', // Canary Islands + '51*', // Ceuta + '52*', // Melilla + ), + 'GR' => array( + '63086', // Mount Athos + '63087', // Mount Athos + ), + 'IT' => array( + '22060', // Livigno, Campione d’Italia + '23030', // Lake Lugano + ), + 'FI' => array( + 'AX*', // Åland Islands + ), + 'CY' => array( + '9*', // Northern Cyprus + '5*', // Northern Cyprus + ), + ); + + if ( array_key_exists( $country, $exemptions ) ) { + foreach ( $exemptions[ $country ] as $exempt_postcode ) { + if ( in_array( $exempt_postcode, $postcode_wildcards, true ) ) { + $belongs = false; + break; + } + } + } + } + + return apply_filters( 'woocommerce_gzd_country_belongs_to_eu_customs_area', $belongs, $country, $postcode ); + } + + public static function is_shipping_international( $country, $args = array() ) { + $args = self::parse_location_data( $args ); + /** + * In case the sender country belongs to EU customs area, a third country needs to lie outside of the EU customs area + */ + if ( self::country_belongs_to_eu_customs_area( $args['sender_country'], $args['sender_postcode'] ) ) { + if ( ! self::country_belongs_to_eu_customs_area( $country, $args['postcode'] ) ) { + return true; + } + + return false; + } else { + if ( ! self::is_shipping_domestic( $country, $args ) ) { + return true; + } + + return false; + } + } + + public static function is_shipping_domestic( $country, $args = array() ) { + $args = self::parse_location_data( $args ); + $is_domestic = $country === $args['sender_country']; + + /** + * If the sender country belongs to EU customs area but the postcode (e.g. Helgoland in DE) not, do not consider domestic shipping + */ + if ( $is_domestic && self::country_belongs_to_eu_customs_area( $args['sender_country'], $args['sender_postcode'] ) ) { + if ( ! self::country_belongs_to_eu_customs_area( $country, $args['postcode'] ) ) { + $is_domestic = false; + } + } + + return $is_domestic; + } + + private static function parse_location_data( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'postcode' => '', + 'sender_country' => self::get_base_country(), + 'sender_postcode' => self::get_base_postcode(), + ) + ); + + return $args; + } + + /** + * Whether shipping is inner EU (from one EU country to another) shipment or not. + * + * @param $country + * @param array $args + * + * @return mixed|void + */ + public static function is_shipping_inner_eu_country( $country, $args = array() ) { + $args = self::parse_location_data( $args ); + + if ( self::is_shipping_domestic( $country, $args ) || ! self::country_belongs_to_eu_customs_area( $args['sender_country'], $args['sender_postcode'] ) ) { + return false; + } + + return self::country_belongs_to_eu_customs_area( $country, $args['postcode'] ); + } + + public static function get_endpoints() { + return array( + 'view-shipment', + 'add-return-shipment', + 'view-shipments', + ); + } + + public static function register_endpoints( $query_vars ) { + foreach ( self::get_endpoints() as $endpoint ) { + if ( ! array_key_exists( $endpoint, $query_vars ) ) { + $option_name = str_replace( '-', '_', $endpoint ); + $query_vars[ $endpoint ] = get_option( "woocommerce_gzd_shipments_{$option_name}_endpoint", $endpoint ); + } + } + + return $query_vars; + } + + public static function install() { + self::init(); + Install::install(); + } + + public static function install_integration() { + self::init(); + self::install(); + } + + public static function maybe_set_upload_dir() { + // Create a dir suffix + if ( ! get_option( 'woocommerce_gzd_shipments_upload_dir_suffix', false ) ) { + self::$upload_dir_suffix = substr( self::generate_key(), 0, 10 ); + update_option( 'woocommerce_gzd_shipments_upload_dir_suffix', self::$upload_dir_suffix ); + } else { + self::$upload_dir_suffix = get_option( 'woocommerce_gzd_shipments_upload_dir_suffix' ); + } + } + + /** + * Generate a unique key. + * + * @return string + */ + protected static function generate_key() { + $key = array( ABSPATH, time() ); + $constants = array( 'AUTH_KEY', 'SECURE_AUTH_KEY', 'LOGGED_IN_KEY', 'NONCE_KEY', 'AUTH_SALT', 'SECURE_AUTH_SALT', 'LOGGED_IN_SALT', 'NONCE_SALT', 'SECRET_KEY' ); + + foreach ( $constants as $constant ) { + if ( defined( $constant ) ) { + $key[] = constant( $constant ); + } + } + + shuffle( $key ); + + return md5( serialize( $key ) ); // phpcs:ignore WordPress.PHP.DiscouragedPHPFunctions.serialize_serialize + } + + public static function log( $message, $type = 'info' ) { + $enable_logging = defined( 'WP_DEBUG' ) && WP_DEBUG ? true : false; + + /** + * Filter that allows adjusting whether to enable or disable + * logging for the shipments package + * + * @param boolean $enable_logging True if logging should be enabled. False otherwise. + * + * @package Vendidero/Germanized/Shipments + */ + if ( ! apply_filters( 'woocommerce_gzd_shipments_enable_logging', $enable_logging ) ) { + return false; + } + + $logger = wc_get_logger(); + + if ( ! $logger ) { + return false; + } + + if ( ! is_callable( array( $logger, $type ) ) ) { + $type = 'info'; + } + + $logger->{$type}( $message, array( 'source' => 'wc-gzd-shipments' ) ); + } + + public static function get_upload_dir_suffix() { + return self::$upload_dir_suffix; + } + + public static function get_upload_dir() { + + self::set_upload_dir_filter(); + $upload_dir = wp_upload_dir(); + self::unset_upload_dir_filter(); + + /** + * Filter to adjust the upload directory used to store shipment related files. By default + * files are stored in a custom directory under wp-content/uploads. + * + * @param array $upload_dir Array containing `wp_upload_dir` data. + * + * @since 3.0.1 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipments_upload_dir', $upload_dir ); + } + + public static function get_relative_upload_dir( $path ) { + + self::set_upload_dir_filter(); + $path = _wp_relative_upload_path( $path ); + self::unset_upload_dir_filter(); + + /** + * Filter to retrieve the relative upload path used for storing shipment related files. + * + * @param array $path Relative path. + * + * @since 3.0.1 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipments_relative_upload_dir', $path ); + } + + public static function set_upload_dir_filter() { + add_filter( 'upload_dir', array( __CLASS__, 'filter_upload_dir' ), 150, 1 ); + } + + public static function unset_upload_dir_filter() { + remove_filter( 'upload_dir', array( __CLASS__, 'filter_upload_dir' ), 150 ); + } + + public static function get_file_by_path( $file ) { + // If the file is relative, prepend upload dir. + if ( $file && 0 !== strpos( $file, '/' ) && ( ( $uploads = self::get_upload_dir() ) && false === $uploads['error'] ) ) { + $file = $uploads['basedir'] . "/$file"; + + return $file; + } else { + return $file; + } + } + + public static function filter_upload_dir( $args ) { + $upload_base = trailingslashit( $args['basedir'] ); + $upload_url = trailingslashit( $args['baseurl'] ); + + /** + * Filter to adjust the upload path used to store shipment related files. By default + * files are stored in a custom directory under wp-content/uploads. + * + * @param string $path Path to the upload directory. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + $args['basedir'] = apply_filters( 'woocommerce_gzd_shipments_upload_path', $upload_base . 'wc-gzd-shipments-' . self::get_upload_dir_suffix() ); + /** + * Filter to adjust the upload URL used to retrieve shipment related files. By default + * files are stored in a custom directory under wp-content/uploads. + * + * @param string $url URL to the upload directory. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + $args['baseurl'] = apply_filters( 'woocommerce_gzd_shipments_upload_url', $upload_url . 'wc-gzd-shipments-' . self::get_upload_dir_suffix() ); + + $args['path'] = $args['basedir'] . $args['subdir']; + $args['url'] = $args['baseurl'] . $args['subdir']; + + return $args; + } + + public static function has_dependencies() { + return class_exists( 'WooCommerce' ) && apply_filters( 'woocommerce_gzd_shipments_enabled', true ); + } + + private static function includes() { + if ( is_admin() ) { + Admin\Admin::init(); + } + + Ajax::init(); + Automation::init(); + Labels\Automation::init(); + Labels\DownloadHandler::init(); + Emails::init(); + Validation::init(); + Api::init(); + FormHandler::init(); + ReportHelper::init(); + + if ( self::is_frontend_request() ) { + include_once self::get_path() . '/includes/wc-gzd-shipment-template-hooks.php'; + } + + include_once self::get_path() . '/includes/wc-gzd-shipment-functions.php'; + include_once self::get_path() . '/includes/wc-gzd-label-functions.php'; + include_once self::get_path() . '/includes/wc-gzd-packaging-functions.php'; + } + + private static function is_frontend_request() { + return ( ! is_admin() || defined( 'DOING_AJAX' ) ) && ! defined( 'DOING_CRON' ); + } + + /** + * Function used to Init WooCommerce Template Functions - This makes them pluggable by plugins and themes. + */ + public static function include_template_functions() { + include_once self::get_path() . '/includes/wc-gzd-shipments-template-functions.php'; + } + + public static function filter_templates( $path, $template_name ) { + + if ( file_exists( self::get_path() . '/templates/' . $template_name ) ) { + $path = self::get_path() . '/templates/' . $template_name; + } + + return $path; + } + + /** + * Register custom tables within $wpdb object. + */ + private static function define_tables() { + global $wpdb; + + // List of tables without prefixes. + $tables = array( + 'gzd_shipment_itemmeta' => 'woocommerce_gzd_shipment_itemmeta', + 'gzd_shipmentmeta' => 'woocommerce_gzd_shipmentmeta', + 'gzd_shipments' => 'woocommerce_gzd_shipments', + 'gzd_shipment_labelmeta' => 'woocommerce_gzd_shipment_labelmeta', + 'gzd_shipment_labels' => 'woocommerce_gzd_shipment_labels', + 'gzd_shipment_items' => 'woocommerce_gzd_shipment_items', + 'gzd_shipping_provider' => 'woocommerce_gzd_shipping_provider', + 'gzd_shipping_providermeta' => 'woocommerce_gzd_shipping_providermeta', + 'gzd_packaging' => 'woocommerce_gzd_packaging', + 'gzd_packagingmeta' => 'woocommerce_gzd_packagingmeta', + ); + + foreach ( $tables as $name => $table ) { + $wpdb->$name = $wpdb->prefix . $table; + $wpdb->tables[] = $table; + } + } + + public static function register_data_stores( $stores ) { + $stores['shipment'] = 'Vendidero\Germanized\Shipments\DataStores\Shipment'; + $stores['shipment-label'] = 'Vendidero\Germanized\Shipments\DataStores\Label'; + $stores['packaging'] = 'Vendidero\Germanized\Shipments\DataStores\Packaging'; + $stores['shipment-item'] = 'Vendidero\Germanized\Shipments\DataStores\ShipmentItem'; + $stores['shipping-provider'] = 'Vendidero\Germanized\Shipments\DataStores\ShippingProvider'; + + do_action( 'woocommerce_gzd_shipments_registered_data_stores' ); + + return $stores; + } + + /** + * Return the version of the package. + * + * @return string + */ + public static function get_version() { + return self::VERSION; + } + + /** + * Return the path to the package. + * + * @return string + */ + public static function get_path() { + return dirname( __DIR__ ); + } + + /** + * Return the path to the package. + * + * @return string + */ + public static function get_url() { + return plugins_url( '', __DIR__ ); + } + + public static function get_assets_url() { + return self::get_url() . '/assets'; + } + + public static function get_setting( $name, $default = false ) { + $option_name = "woocommerce_gzd_shipments_{$name}"; + + return get_option( $option_name, $default ); + } + + public static function get_store_address_country() { + $default = get_option( 'woocommerce_store_country' ); + + return $default; + } + + public static function get_store_address_street() { + $store_address = wc_gzd_split_shipment_street( get_option( 'woocommerce_store_address' ) ); + + return $store_address['street']; + } + + public static function get_store_address_street_number() { + $store_address = wc_gzd_split_shipment_street( get_option( 'woocommerce_store_address' ) ); + + return $store_address['number']; + } + + public static function is_valid_datetime( $maybe_datetime, $format = 'Y-m-d' ) { + if ( ! is_a( $maybe_datetime, 'DateTime' && ! is_numeric( $maybe_datetime ) ) ) { + if ( ! \DateTime::createFromFormat( $format, $maybe_datetime ) ) { + return false; + } + } + + return true; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packaging.php b/packages/woocommerce-germanized-shipments/src/Packaging.php new file mode 100644 index 000000000..232881501 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packaging.php @@ -0,0 +1,339 @@ + null, + 'weight' => 0, + 'max_content_weight' => 0, + 'width' => 0, + 'height' => 0, + 'length' => 0, + 'order' => 0, + 'type' => '', + 'description' => '', + ); + + /** + * Get the packaging if ID is passed, otherwise the packaging is new and empty. + * This class should NOT be instantiated, but the `wc_gzd_get_packaging` function should be used. + * + * @param int|object|Packaging $packaging packaging to read. + */ + public function __construct( $data = 0 ) { + parent::__construct( $data ); + + if ( $data instanceof Packaging ) { + $this->set_id( absint( $data->get_id() ) ); + } elseif ( is_numeric( $data ) ) { + $this->set_id( $data ); + } + + $this->data_store = WC_Data_Store::load( $this->data_store_name ); + + // If we have an ID, load the user from the DB. + if ( $this->get_id() ) { + try { + $this->data_store->read( $this ); + } catch ( Exception $e ) { + $this->set_id( 0 ); + $this->set_object_read( true ); + } + } else { + $this->set_object_read( true ); + } + } + + /** + * Merge changes with data and clear. + * Overrides WC_Data::apply_changes. + * + * @since 3.2.0 + */ + public function apply_changes() { + if ( function_exists( 'array_replace' ) ) { + $this->data = array_replace( $this->data, $this->changes ); // phpcs:ignore PHPCompatibility.FunctionUse.NewFunctions.array_replaceFound + } else { // PHP 5.2 compatibility. + foreach ( $this->changes as $key => $change ) { + $this->data[ $key ] = $change; + } + } + $this->changes = array(); + } + + /** + * Prefix for action and filter hooks on data. + * + * @return string + */ + protected function get_hook_prefix() { + return $this->get_general_hook_prefix() . 'get_'; + } + + /** + * Prefix for action and filter hooks on data. + * + * @return string + */ + protected function get_general_hook_prefix() { + return 'woocommerce_gzd_packaging_'; + } + + /** + * Return the date this packaging was created. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return WC_DateTime|null object if the date is set or null if there is no date. + */ + public function get_date_created( $context = 'view' ) { + return $this->get_prop( 'date_created', $context ); + } + + /** + * Returns the packaging weight in kg. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_weight( $context = 'view' ) { + return $this->get_prop( 'weight', $context ); + } + + /** + * Returns the packaging max content weight in kg. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_max_content_weight( $context = 'view' ) { + return $this->get_prop( 'max_content_weight', $context ); + } + + /** + * Returns the packaging order within its list. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_order( $context = 'view' ) { + return $this->get_prop( 'order', $context ); + } + + /** + * Returns the packaging type e.g. box or letter. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_type( $context = 'view' ) { + return $this->get_prop( 'type', $context ); + } + + /** + * Returns the packaging description. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_description( $context = 'view' ) { + return $this->get_prop( 'description', $context ); + } + + /** + * Returns the packaging length in cm. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_length( $context = 'view' ) { + return $this->get_prop( 'length', $context ); + } + + /** + * Returns the packaging width in cm. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_width( $context = 'view' ) { + return $this->get_prop( 'width', $context ); + } + + /** + * Returns the packaging height in cm. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_height( $context = 'view' ) { + return $this->get_prop( 'height', $context ); + } + + public function has_dimensions() { + $width = $this->get_width(); + $length = $this->get_length(); + $height = $this->get_height(); + + return ( ! empty( $width ) && ! empty( $length ) && ! empty( $height ) ); + } + + /** + * Returns dimensions. + * + * @return string|array + */ + public function get_dimensions() { + return array( + 'length' => wc_format_decimal( $this->get_length(), false, true ), + 'width' => wc_format_decimal( $this->get_width(), false, true ), + 'height' => wc_format_decimal( $this->get_height(), false, true ), + ); + } + + public function get_formatted_dimensions() { + return wc_gzd_format_shipment_dimensions( $this->get_dimensions(), wc_gzd_get_packaging_dimension_unit() ); + } + + public function get_volume() { + return (float) $this->get_length() * (float) $this->get_width() * (float) $this->get_height(); + } + + /** + * Set the date this packaging was created. + * + * @param string|integer|null $date UTC timestamp, or ISO 8601 DateTime. If the DateTime string has no timezone or offset, WordPress site timezone will be assumed. Null if their is no date. + */ + public function set_date_created( $date = null ) { + $this->set_date_prop( 'date_created', $date ); + } + + /** + * Set packaging weight in kg. + * + * @param string $weight The weight. + */ + public function set_weight( $weight ) { + $this->set_prop( 'weight', empty( $weight ) ? 0 : wc_format_decimal( $weight, 2, true ) ); + } + + public function get_title() { + $description = $this->get_description(); + + return sprintf( + _x( '%1$s (%2$s, %3$s)', 'shipments-packaging-title', 'woocommerce-germanized' ), + $description, + $this->get_formatted_dimensions(), + wc_gzd_format_shipment_weight( wc_format_decimal( $this->get_weight(), false, true ), wc_gzd_get_packaging_weight_unit() ) + ); + } + + /** + * Set packaging order. + * + * @param integer $order The order. + */ + public function set_order( $order ) { + $this->set_prop( 'order', absint( $order ) ); + } + + /** + * Set packaging max content weight in kg. + * + * @param string $weight The weight. + */ + public function set_max_content_weight( $weight ) { + $this->set_prop( 'max_content_weight', empty( $weight ) ? 0 : wc_format_decimal( $weight, 2, true ) ); + } + + /** + * Set packaging type + * + * @param string $type The type. + */ + public function set_type( $type ) { + $this->set_prop( 'type', $type ); + } + + /** + * Set packaging description + * + * @param string $description The description. + */ + public function set_description( $description ) { + $this->set_prop( 'description', $description ); + } + + /** + * Set packaging width in cm. + * + * @param string $width The width. + */ + public function set_width( $width ) { + $this->set_prop( 'width', empty( $width ) ? 0 : wc_format_decimal( $width, 1, true ) ); + } + + /** + * Set packaging length in cm. + * + * @param string $length The length. + */ + public function set_length( $length ) { + $this->set_prop( 'length', empty( $length ) ? 0 : wc_format_decimal( $length, 1, true ) ); + } + + /** + * Set packaging height in cm. + * + * @param string $height The height. + */ + public function set_height( $height ) { + $this->set_prop( 'height', empty( $height ) ? 0 : wc_format_decimal( $height, 1, true ) ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packaging/AsyncReportGenerator.php b/packages/woocommerce-germanized-shipments/src/Packaging/AsyncReportGenerator.php new file mode 100644 index 000000000..2fe1db778 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packaging/AsyncReportGenerator.php @@ -0,0 +1,195 @@ +type = $type; + $default_end = new \WC_DateTime(); + $default_start = new \WC_DateTime( 'now' ); + $default_start->modify( '-1 year' ); + + $args = wp_parse_args( + $args, + array( + 'start' => $default_start->format( 'Y-m-d' ), + 'end' => $default_end->format( 'Y-m-d' ), + 'limit' => ReportQueue::get_batch_size(), + 'status' => ReportQueue::get_shipment_statuses(), + 'offset' => 0, + 'type' => apply_filters( 'woocommerce_gzd_shipments_packaging_report_default_shipment_types', array( 'simple' ) ), + 'processed' => 0, + ) + ); + + foreach ( array( 'start', 'end' ) as $date_field ) { + if ( is_a( $args[ $date_field ], 'WC_DateTime' ) ) { + $args[ $date_field ] = $args[ $date_field ]->format( 'Y-m-d' ); + } elseif ( is_numeric( $args[ $date_field ] ) ) { + $date = new \WC_DateTime( '@' . $args[ $date_field ] ); + $args[ $date_field ] = $date->format( 'Y-m-d' ); + } + } + + $this->args = $args; + } + + public function get_type() { + return $this->type; + } + + public function get_args() { + return $this->args; + } + + public function get_id() { + return ReportHelper::get_report_id( + array( + 'type' => $this->type, + 'date_start' => $this->args['start'], + 'date_end' => $this->args['end'], + ) + ); + } + + public function delete() { + $report = new Report( $this->get_id() ); + $report->delete(); + + delete_option( $this->get_id() . '_tmp_result' ); + } + + public function start() { + $report = new Report( $this->get_id() ); + $report->reset(); + $report->save(); + + return $report; + } + + /** + * @return true|\WP_Error + */ + public function next() { + $args = $this->args; + $shipments = ReportQueue::query( $args ); + $shipments_processed = 0; + $packaging_data = $this->get_temporary_result(); + + Package::log( sprintf( '%d applicable shipments found', count( $shipments ) ) ); + + if ( ! empty( $shipments ) ) { + foreach ( $shipments as $shipment ) { + $packaging_weight = $shipment->get_packaging_weight(); + $packaging = $shipment->get_packaging(); + $country = $shipment->get_country(); + $packaging_id = 'other'; + + if ( ! $shipment->get_packaging_weight() ) { + Package::log( sprintf( 'Skipping shipment #%1$s due to missing packaging weight.', $shipment->get_id() ) ); + continue; + } + + if ( $packaging ) { + $packaging_id = $packaging->get_id(); + } + + if ( ! isset( $packaging_data[ $country ][ "$packaging_id" ] ) ) { + $packaging_data[ $country ][ "$packaging_id" ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + ); + } + + $packaging_data[ $country ][ "$packaging_id" ]['count'] += 1; + $packaging_data[ $country ][ "$packaging_id" ]['weight_in_kg'] += wc_add_number_precision( (float) wc_get_weight( $packaging_weight, 'kg', $shipment->get_weight_unit() ), false ); + + $shipments_processed++; + } + + $this->args['processed'] = absint( $this->args['processed'] ) + $shipments_processed; + + update_option( $this->get_id() . '_tmp_result', $packaging_data, false ); + + return true; + } else { + return new \WP_Error( 'empty', _x( 'No shipments found.', 'shipments', 'woocommerce-germanized' ) ); + } + } + + /** + * @return Report + */ + public function complete() { + Package::log( sprintf( 'Completed called' ) ); + + $tmp_result = $this->get_temporary_result(); + $report = new Report( $this->get_id() ); + $count_total = 0; + $weight_total = 0.0; + $packaging_totals = array(); + + foreach ( $tmp_result as $country => $packaging_ids ) { + $country_count_total = 0; + $country_weight_total = 0.0; + + foreach ( $packaging_ids as $packaging_id => $totals ) { + $count_total += (int) $totals['count']; + $weight_total += (float) $totals['weight_in_kg']; + $country_count_total += (int) $totals['count']; + $country_weight_total += (float) $totals['weight_in_kg']; + + if ( ! isset( $packaging_totals[ "$packaging_id" ] ) ) { + $packaging_totals[ "$packaging_id" ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + ); + } + + $packaging_totals[ "$packaging_id" ]['count'] += (int) $totals['count']; + $packaging_totals[ "$packaging_id" ]['weight_in_kg'] += (float) $totals['weight_in_kg']; + + $report->set_packaging_count_by_country( $country, $packaging_id, (int) $totals['count'] ); + $report->set_packaging_weight_by_country( $country, $packaging_id, (float) wc_remove_number_precision( $totals['weight_in_kg'] ) ); + } + + $country_weight_total = (float) wc_remove_number_precision( $country_weight_total ); + + $report->set_total_packaging_count_by_country( $country, $country_count_total ); + $report->set_total_packaging_weight_by_country( $country, $country_weight_total ); + } + + foreach ( $packaging_totals as $packaging_id => $totals ) { + $packaging_weight_total = (float) wc_remove_number_precision( $totals['weight_in_kg'] ); + + $report->set_packaging_count( $packaging_id, $totals['count'] ); + $report->set_packaging_weight( $packaging_id, $packaging_weight_total ); + } + + $weight_total = (float) wc_remove_number_precision( $weight_total ); + + Package::log( sprintf( 'Completed packaging count: %d', $count_total ) ); + Package::log( sprintf( 'Completed packaging weight: %s', $weight_total ) ); + + $report->set_total_count( $count_total ); + $report->set_total_weight( $weight_total ); + $report->set_status( 'completed' ); + $report->set_version( Package::get_version() ); + $report->save(); + + return $report; + } + + protected function get_temporary_result() { + return (array) get_option( $this->get_id() . '_tmp_result', array() ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packaging/Report.php b/packages/woocommerce-germanized-shipments/src/Packaging/Report.php new file mode 100644 index 000000000..099b06851 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packaging/Report.php @@ -0,0 +1,422 @@ +set_id( $id ); + + if ( empty( $args ) ) { + $args = (array) get_option( $this->id . '_result', array() ); + } + + $args = wp_parse_args( + $args, + array( + 'packaging' => array(), + 'countries' => array(), + 'totals' => array(), + 'meta' => array(), + ) + ); + + $args['totals'] = wp_parse_args( + $args['totals'], + array( + 'weight_in_kg' => 0.0, + 'count' => 0, + ) + ); + + $args['meta'] = wp_parse_args( + $args['meta'], + array( + 'date_requested' => null, + 'status' => 'pending', + 'version' => '', + ) + ); + + $this->set_date_requested( $args['meta']['date_requested'] ); + $this->set_status( $args['meta']['status'] ); + $this->set_version( $args['meta']['version'] ); + + $this->args = $args; + } + + public function exists() { + return get_option( $this->id . '_result', false ); + } + + public function get_title() { + $title = ReportHelper::get_report_title( $this->get_id() ); + + if ( $this->get_date_requested() ) { + $title = $title . ' @ ' . $this->get_date_requested()->date_i18n(); + } + + return $title; + } + + public function get_url() { + return admin_url( 'admin.php?page=shipment-packaging-report&report=' . $this->get_id() ); + } + + public function get_delete_link() { + return add_query_arg( + array( + 'action' => 'wc_gzd_shipments_packaging_delete_report', + 'report_id' => $this->get_id(), + ), + wp_nonce_url( admin_url( 'admin-post.php' ), 'wc_gzd_shipments_packaging_delete_report' ) + ); + } + + public function get_refresh_link() { + return add_query_arg( + array( + 'action' => 'wc_gzd_shipments_packaging_refresh_report', + 'report_id' => $this->get_id(), + ), + wp_nonce_url( admin_url( 'admin-post.php' ), 'wc_gzd_shipments_packaging_refresh_report' ) + ); + } + + public function get_cancel_link() { + return add_query_arg( + array( + 'action' => 'wc_gzd_shipments_packaging_cancel_report', + 'report_id' => $this->get_id(), + ), + wp_nonce_url( admin_url( 'admin-post.php' ), 'wc_gzd_shipments_packaging_cancel_report' ) + ); + } + + public function get_type() { + return $this->type; + } + + public function set_type( $type ) { + $this->set_id_part( $type, 'type' ); + } + + public function set_id( $id ) { + $this->id = $id; + $data = ReportHelper::get_report_data( $this->id ); + $this->type = $data['type']; + $this->date_start = $data['date_start']; + $this->date_end = $data['date_end']; + } + + public function set_id_part( $value, $part = 'type' ) { + $data = ReportHelper::get_report_data( $this->id ); + $data[ $part ] = $value; + + $this->set_id( ReportHelper::get_report_id( $data ) ); + } + + public function get_id() { + return $this->id; + } + + public function get_date_start() { + return $this->date_start; + } + + public function set_date_start( $date ) { + $date = ReportHelper::string_to_datetime( $date ); + + $this->set_id_part( $date->format( 'Y-m-d' ), 'date_start' ); + } + + public function get_date_end() { + return $this->date_end; + } + + public function set_date_end( $date ) { + $date = ReportHelper::string_to_datetime( $date ); + + $this->set_id_part( $date->format( 'Y-m-d' ), 'date_end' ); + } + + public function get_status() { + return $this->args['meta']['status']; + } + + public function get_version() { + return $this->args['meta']['version']; + } + + public function set_status( $status ) { + $this->args['meta']['status'] = $status; + } + + public function set_version( $version ) { + $this->args['meta']['version'] = $version; + } + + public function get_date_requested() { + return is_null( $this->args['meta']['date_requested'] ) ? null : ReportHelper::string_to_datetime( $this->args['meta']['date_requested'] ); + } + + public function set_date_requested( $date ) { + if ( ! empty( $date ) ) { + $date = ReportHelper::string_to_datetime( $date ); + } + + $this->args['meta']['date_requested'] = is_a( $date, 'WC_DateTime' ) ? $date->date( 'Y-m-d' ) : null; + } + + /** + * @return int + */ + public function get_total_count() { + return (int) $this->args['totals']['count']; + } + + public function get_total_weight( $round = true, $unit = '' ) { + if ( '' === $unit ) { + $unit = wc_gzd_get_packaging_weight_unit(); + } + + $weight = wc_get_weight( $this->args['totals']['weight_in_kg'], $unit, 'kg' ); + + return $this->maybe_round( $weight, $round ); + } + + public function set_total_weight( $weight ) { + $this->args['totals']['weight_in_kg'] = wc_format_decimal( floatval( $weight ) ); + } + + public function set_total_count( $count ) { + $this->args['totals']['count'] = absint( $count ); + } + + public function get_packaging_ids() { + return array_keys( $this->args['packaging'] ); + } + + public function get_countries() { + return array_keys( $this->args['countries'] ); + } + + public function reset() { + $this->args['packaging'] = array(); + $this->args['countries'] = array(); + + $this->set_total_count( 0 ); + $this->set_total_weight( 0 ); + $this->set_date_requested( new \WC_DateTime() ); + $this->set_status( 'pending' ); + $this->set_version( Package::get_version() ); + + delete_option( $this->id . '_tmp_result' ); + } + + public function get_packaging_ids_by_country( $country ) { + $packaging_ids = array(); + + if ( array_key_exists( $country, $this->args['countries'] ) ) { + $packaging_ids = array_keys( $this->args['countries'][ $country ]['packaging'] ); + } + + return $packaging_ids; + } + + public function get_packaging_count( $packaging_id, $country = '' ) { + $count = 0; + + if ( '' === $country ) { + if ( isset( $this->args['packaging'][ "$packaging_id" ] ) ) { + $count = absint( $this->args['packaging'][ "$packaging_id" ]['count'] ); + } + } else { + if ( isset( $this->args['countries'][ $country ], $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ] ) ) { + $count = absint( $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ]['count'] ); + } + } + + return $count; + } + + public function get_packaging_weight( $packaging_id, $country = '', $round = true, $unit = '' ) { + $weight = 0.0; + + if ( '' === $country ) { + if ( isset( $this->args['packaging'][ "$packaging_id" ] ) ) { + $weight = $this->args['packaging'][ "$packaging_id" ]['weight_in_kg']; + } + } else { + if ( isset( $this->args['countries'][ $country ], $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ] ) ) { + $weight = $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ]['weight_in_kg']; + } + } + + $weight = wc_get_weight( $weight, $unit, 'kg' ); + + return $this->maybe_round( $weight, $round ); + } + + public function get_total_packaging_weight_by_country( $country, $round = true, $unit = '' ) { + $weight = 0.0; + + if ( isset( $this->args['countries'][ $country ] ) ) { + $weight = $this->args['countries'][ $country ]['weight_in_kg']; + } + + $weight = wc_get_weight( $weight, $unit, 'kg' ); + + return $this->maybe_round( $weight, $round ); + } + + public function get_total_packaging_count_by_country( $country ) { + $count = 0; + + if ( isset( $this->args['countries'][ $country ] ) ) { + $count = absint( $this->args['countries'][ $country ]['count'] ); + } + + return $count; + } + + public function set_packaging_count( $packaging_id, $count ) { + if ( ! isset( $this->args['packaging'][ "$packaging_id" ] ) ) { + $this->args['packaging'][ "$packaging_id" ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + ); + } + + $this->args['packaging'][ "$packaging_id" ]['count'] = absint( $count ); + } + + public function set_packaging_weight( $packaging_id, $weight ) { + if ( ! isset( $this->args['packaging'][ "$packaging_id" ] ) ) { + $this->args['packaging'][ "$packaging_id" ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + ); + } + + $this->args['packaging'][ "$packaging_id" ]['weight_in_kg'] = (float) wc_format_decimal( $weight ); + } + + public function set_packaging_count_by_country( $country, $packaging_id, $count ) { + if ( ! isset( $this->args['countries'][ $country ] ) ) { + $this->args['countries'][ $country ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + 'packaging' => array(), + ); + } + + if ( ! isset( $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ] ) ) { + $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + ); + } + + $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ]['count'] = absint( $count ); + } + + public function set_packaging_weight_by_country( $country, $packaging_id, $weight ) { + if ( ! isset( $this->args['countries'][ $country ] ) ) { + $this->args['countries'][ $country ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + 'packaging' => array(), + ); + } + + if ( ! isset( $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ] ) ) { + $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + ); + } + + $this->args['countries'][ $country ]['packaging'][ "$packaging_id" ]['weight_in_kg'] = (float) wc_format_decimal( $weight ); + } + + public function set_total_packaging_count_by_country( $country, $count ) { + if ( ! isset( $this->args['countries'][ $country ] ) ) { + $this->args['countries'][ $country ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + 'packaging' => array(), + ); + } + + $this->args['countries'][ $country ]['count'] = absint( $count ); + } + + public function set_total_packaging_weight_by_country( $country, $weight ) { + if ( ! isset( $this->args['countries'][ $country ] ) ) { + $this->args['countries'][ $country ] = array( + 'count' => 0, + 'weight_in_kg' => 0.0, + 'packaging' => array(), + ); + } + + $this->args['countries'][ $country ]['weight_in_kg'] = (float) wc_format_decimal( $weight ); + } + + protected function maybe_round( $total, $round = true ) { + $decimals = is_numeric( $round ) ? (int) $round : ''; + + return (float) wc_format_decimal( $total, $round ? $decimals : false ); + } + + public function save() { + update_option( $this->id . '_result', $this->args, false ); + + $reports_available = ReportHelper::get_report_ids(); + + if ( ! in_array( $this->get_id(), $reports_available[ $this->get_type() ], true ) ) { + // Add new report to start of the list + array_unshift( $reports_available[ $this->get_type() ], $this->get_id() ); + update_option( 'woocommerce_gzd_shipments_packaging_reports', $reports_available, false ); + } + + delete_option( $this->id . '_tmp_result' ); + + ReportHelper::clear_caches(); + + return $this->id; + } + + public function delete() { + delete_option( $this->id . '_result' ); + delete_option( $this->id . '_tmp_result' ); + + ReportQueue::maybe_stop_report( $this->get_id() ); + ReportHelper::remove_report( $this ); + + ReportHelper::clear_caches(); + + return true; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packaging/ReportHelper.php b/packages/woocommerce-germanized-shipments/src/Packaging/ReportHelper.php new file mode 100644 index 000000000..5eef695a2 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packaging/ReportHelper.php @@ -0,0 +1,516 @@ + array( + 'url' => $report->get_url(), + 'title' => _x( 'View', 'shipments-packaging-report', 'woocommerce-germanized' ), + ), + 'refresh' => array( + 'url' => $report->get_refresh_link(), + 'title' => _x( 'Refresh', 'shipments-packaging-report', 'woocommerce-germanized' ), + ), + 'delete' => array( + 'url' => $report->get_delete_link(), + 'title' => _x( 'Delete', 'shipments-packaging-report', 'woocommerce-germanized' ), + ), + ); + + if ( 'completed' !== $report->get_status() ) { + $actions['cancel'] = $actions['delete']; + $actions['cancel']['title'] = _x( 'Cancel', 'shipments-packaging-report', 'woocommerce-germanized' ); + + unset( $actions['view'] ); + unset( $actions['refresh'] ); + unset( $actions['delete'] ); + } + + return $actions; + } + + public static function render_report() { + if ( isset( $_GET['report'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $report_id = isset( $_GET['report'] ) ? wc_clean( wp_unslash( $_GET['report'] ) ) : false; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + if ( ! $report_id ) { + return; + } + + if ( ! $report = self::get_report( $report_id ) ) { + return; + } + + $columns = array( + 'packaging' => _x( 'Packaging', 'shipments', 'woocommerce-germanized' ), + 'weight' => _x( 'Weight', 'shipments', 'woocommerce-germanized' ), + 'count' => _x( 'Count', 'shipments', 'woocommerce-germanized' ), + ); + + $actions = self::get_report_actions( $report ); + $countries = WC()->countries->get_countries(); + $packaging_ids = $report->get_packaging_ids(); + ?> +
+

get_title() ); ?>

+ $action ) : + if ( 'view' === $action_type ) { + continue; + } + ?> + + + + get_status() ) : ?> +

get_date_start()->date_i18n( wc_date_format() ) ); ?> – get_date_end()->date_i18n( wc_date_format() ) ); ?>: get_total_weight(), wc_gzd_get_packaging_weight_unit() ) ); ?> (get_total_count() ) ); ?>)

+
+ + + + + + $column ) : ?> + + + + + + + + + + + + + +
get_id() > 0 ? $packaging->get_description() : _x( 'Unknown', 'shipments-packaging-title', 'woocommerce-germanized' ) ) ); ?>get_packaging_weight( $packaging_id ), wc_gzd_get_packaging_weight_unit() ) ); ?>get_packaging_count( $packaging_id ) ); ?>
+ + + get_countries() as $country ) : + ?> +

+

get_total_packaging_weight_by_country( $country ), wc_gzd_get_packaging_weight_unit() ) ); ?> (get_total_packaging_count_by_country( $country ) ) ); ?>)

+ + + + $column ) : ?> + + + + + + get_packaging_ids_by_country( $country ) as $packaging_id ) : + $packaging = is_numeric( $packaging_id ) ? wc_gzd_get_packaging( $packaging_id ) : false; + ?> + + + + + + + +
get_id() > 0 ? $packaging->get_description() : _x( 'Unknown', 'shipments-packaging-title', 'woocommerce-germanized' ) ) ); ?>get_packaging_weight( $packaging_id, $country ), wc_gzd_get_packaging_weight_unit() ) ); ?>get_packaging_count( $packaging_id, $country ) ); ?>
+ + +

Find pending actions', 'shipments', 'woocommerce-germanized' ), esc_html( $details['shipment_count'] ), ( $details['next_date'] ? esc_html( $details['next_date']->date_i18n( wc_date_format() . ' @ ' . wc_time_format() ) ) : esc_html_x( 'Not yet known', 'shipments', 'woocommerce-germanized' ) ), esc_url( $details['link'] ) ); ?>

+ +
+ get_status() ) { + $report->delete(); + } + } + } + } + + $running = array_values( $running ); + + update_option( 'woocommerce_gzd_shipments_packaging_reports_running', $running, false ); + ReportQueue::clear_cache(); + } + + public static function setup_recurring_actions() { + if ( $queue = ReportQueue::get_queue() ) { + // Schedule once per day at 2:00 + if ( null === $queue->get_next( 'woocommerce_gzd_shipments_daily_cleanup', array(), 'woocommerce_gzd_shipments' ) ) { + $timestamp = strtotime( 'tomorrow midnight' ); + $date = new \WC_DateTime(); + + $date->setTimestamp( $timestamp ); + $date->modify( '+2 hours' ); + + $queue->cancel_all( 'woocommerce_gzd_shipments_daily_cleanup', array(), 'woocommerce_gzd_shipments' ); + $queue->schedule_recurring( $date->getTimestamp(), DAY_IN_SECONDS, 'woocommerce_gzd_shipments_daily_cleanup', array(), 'woocommerce_gzd_shipments' ); + } + } + } + + public static function get_report_title( $id ) { + $args = self::get_report_data( $id ); + $title = _x( 'Report', 'shipments', 'woocommerce-germanized' ); + + if ( 'quarterly' === $args['type'] ) { + $date_start = $args['date_start']; + $quarter = 1; + $month_num = (int) $date_start->date_i18n( 'n' ); + + if ( 4 === $month_num ) { + $quarter = 2; + } elseif ( 7 === $month_num ) { + $quarter = 3; + } elseif ( 10 === $month_num ) { + $quarter = 4; + } + + $title = sprintf( _x( 'Q%1$s/%2$s', 'shipments', 'woocommerce-germanized' ), $quarter, $date_start->date_i18n( 'Y' ) ); + } elseif ( 'monthly' === $args['type'] ) { + $date_start = $args['date_start']; + $month_num = $date_start->date_i18n( 'm' ); + + $title = sprintf( _x( '%1$s/%2$s', 'shipments', 'woocommerce-germanized' ), $month_num, $date_start->date_i18n( 'Y' ) ); + } elseif ( 'yearly' === $args['type'] ) { + $date_start = $args['date_start']; + + $title = sprintf( _x( '%1$s', 'shipments', 'woocommerce-germanized' ), $date_start->date_i18n( 'Y' ) ); // phpcs:ignore WordPress.WP.I18n.NoEmptyStrings + } elseif ( 'custom' === $args['type'] ) { + $date_start = $args['date_start']; + $date_end = $args['date_end']; + + $title = sprintf( _x( '%1$s - %2$s', 'shipments', 'woocommerce-germanized' ), $date_start->date_i18n( 'Y-m-d' ), $date_end->date_i18n( 'Y-m-d' ) ); + } + + return $title; + } + + public static function get_report_id( $parts ) { + $parts = wp_parse_args( + $parts, + array( + 'type' => 'daily', + 'date_start' => date( 'Y-m-d' ), // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + 'date_end' => date( 'Y-m-d' ), // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + ) + ); + + if ( is_a( $parts['date_start'], 'WC_DateTime' ) ) { + $parts['date_start'] = $parts['date_start']->format( 'Y-m-d' ); + } + + if ( is_a( $parts['date_end'], 'WC_DateTime' ) ) { + $parts['date_end'] = $parts['date_end']->format( 'Y-m-d' ); + } + + return sanitize_key( 'woocommerce_gzd_shipments_packaging_' . $parts['type'] . '_report_' . $parts['date_start'] . '_' . $parts['date_end'] ); + } + + public static function get_available_report_types() { + $types = array( + 'quarterly' => _x( 'Quarterly', 'shipments', 'woocommerce-germanized' ), + 'yearly' => _x( 'Yearly', 'shipments', 'woocommerce-germanized' ), + 'monthly' => _x( 'Monthly', 'shipments', 'woocommerce-germanized' ), + 'custom' => _x( 'Custom', 'shipments', 'woocommerce-germanized' ), + ); + + return $types; + } + + public static function get_report_data( $id ) { + $clean_id = str_replace( 'packaging_', '', $id ); + $clean_id = str_replace( 'woocommerce_gzd_shipments_', '', $clean_id ); + $id_parts = explode( '_', $clean_id ); + + $data = array( + 'id' => $id, + 'type' => $id_parts[0], + 'date_start' => self::string_to_datetime( $id_parts[2] ), + 'date_end' => self::string_to_datetime( $id_parts[3] ), + ); + + return $data; + } + + public static function string_to_datetime( $time_string ) { + if ( is_string( $time_string ) && ! is_numeric( $time_string ) ) { + $time_string = strtotime( $time_string ); + } + + $date_time = $time_string; + + if ( is_numeric( $date_time ) ) { + $date_time = new \WC_DateTime( "@{$date_time}", new \DateTimeZone( 'UTC' ) ); + } + + if ( ! is_a( $date_time, 'WC_DateTime' ) ) { + return null; + } + + return $date_time; + } + + public static function clear_caches() { + delete_transient( 'woocommerce_gzd_shipments_packaging_report_counts' ); + wp_cache_delete( 'woocommerce_gzd_shipments_packaging_reports', 'options' ); + } + + public static function get_report_ids() { + $reports = (array) get_option( 'woocommerce_gzd_shipments_packaging_reports', array() ); + + foreach ( array_keys( self::get_available_report_types() ) as $type ) { + if ( ! array_key_exists( $type, $reports ) ) { + $reports[ $type ] = array(); + } + } + + return $reports; + } + + public static function get_report_status_title( $status ) { + if ( 'completed' === $status ) { + return _x( 'Completed', 'shipments-report-status', 'woocommerce-germanized' ); + } else { + return _x( 'Pending', 'shipments-report-status', 'woocommerce-germanized' ); + } + } + + /** + * @param array $args + * + * @return Report[] + */ + public static function get_reports( $args = array() ) { + $args = wp_parse_args( + $args, + array( + 'type' => '', + 'limit' => -1, + 'offset' => 0, + 'orderby' => 'date_start', + ) + ); + + $ids = self::get_report_ids(); + + if ( ! empty( $args['type'] ) ) { + $report_ids = array_key_exists( $args['type'], $ids ) ? $ids[ $args['type'] ] : array(); + } else { + $report_ids = array_merge( ...array_values( $ids ) ); + } + + $reports_sorted = array(); + + foreach ( $report_ids as $id ) { + $reports_sorted[] = self::get_report_data( $id ); + } + + if ( array_key_exists( $args['orderby'], array( 'date_start', 'date_end' ) ) ) { + usort( + $reports_sorted, + function( $a, $b ) use ( $args ) { + if ( $a[ $args['orderby'] ] === $b[ $args['orderby'] ] ) { + return 0; + } + + return $a[ $args['orderby'] ] < $b[ $args['orderby'] ] ? -1 : 1; + } + ); + } + + if ( -1 !== $args['limit'] ) { + $reports_sorted = array_slice( $reports_sorted, $args['offset'], $args['limit'] ); + } + + $reports = array(); + + foreach ( $reports_sorted as $data ) { + if ( $report = self::get_report( $data['id'] ) ) { + $reports[] = $report; + } + } + + return $reports; + } + + /** + * @param Report $report + */ + public static function remove_report( $report ) { + $reports_available = self::get_report_ids(); + + if ( in_array( $report->get_id(), $reports_available[ $report->get_type() ], true ) ) { + $reports_available[ $report->get_type() ] = array_diff( $reports_available[ $report->get_type() ], array( $report->get_id() ) ); + + update_option( 'woocommerce_gzd_shipments_packaging_reports', $reports_available, false ); + + /** + * Force non-cached option + */ + wp_cache_delete( 'woocommerce_gzd_shipments_packaging_reports', 'options' ); + } + } + + /** + * @param $id + * + * @return false|Report + */ + public static function get_report( $id ) { + $report = new Report( $id ); + + if ( $report->exists() ) { + return $report; + } + + return false; + } + + public static function delete_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'wc_gzd_shipments_packaging_delete_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_id = isset( $_GET['report_id'] ) ? wc_clean( wp_unslash( $_GET['report_id'] ) ) : ''; + + if ( ! empty( $report_id ) && ( $report = self::get_report( $report_id ) ) ) { + $report->delete(); + + $referer = self::get_clean_referer(); + + /** + * Do not redirect deleted, refreshed reports back to report details page + */ + if ( strstr( $referer, '&report=' ) ) { + $referer = admin_url( 'admin.php?page=wc-settings&tab=germanized-shipments§ion=packaging' ); + } + + wp_safe_redirect( esc_url_raw( add_query_arg( array( 'report_deleted' => $report_id ), $referer ) ) ); + exit(); + } + + wp_safe_redirect( esc_url_raw( wp_get_referer() ) ); + exit(); + } + + protected static function get_clean_referer() { + $referer = wp_get_referer(); + + return remove_query_arg( array( 'report_created', 'report_deleted', 'report_restarted', 'report_cancelled' ), $referer ); + } + + public static function refresh_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'wc_gzd_shipments_packaging_refresh_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_id = isset( $_GET['report_id'] ) ? wc_clean( wp_unslash( $_GET['report_id'] ) ) : ''; + + if ( ! empty( $report_id ) && ( $report = self::get_report( $report_id ) ) ) { + ReportQueue::start( $report->get_type(), $report->get_date_start(), $report->get_date_end() ); + + wp_safe_redirect( esc_url_raw( add_query_arg( array( 'report_restarted' => $report_id ), self::get_clean_referer() ) ) ); + exit(); + } + + wp_safe_redirect( esc_url_raw( wp_get_referer() ) ); + exit(); + } + + public static function cancel_report() { + if ( ! current_user_can( 'manage_woocommerce' ) || ! wp_verify_nonce( isset( $_GET['_wpnonce'] ) ? wp_unslash( $_GET['_wpnonce'] ) : '', 'wc_gzd_shipments_packaging_cancel_report' ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + wp_die(); + } + + $report_id = isset( $_GET['report_id'] ) ? wc_clean( wp_unslash( $_GET['report_id'] ) ) : ''; + + if ( ! empty( $report_id ) && ReportQueue::is_running( $report_id ) ) { + ReportQueue::cancel( $report_id ); + + $referer = self::get_clean_referer(); + + /** + * Do not redirect deleted, refreshed reports back to report details page + */ + if ( strstr( $referer, '&report=' ) ) { + $referer = admin_url( 'admin.php?page=wc-settings&tab=germanized-shipments§ion=packaging' ); + } + + wp_safe_redirect( esc_url_raw( add_query_arg( array( 'report_cancelled' => $report_id ), $referer ) ) ); + exit(); + } + + wp_safe_redirect( esc_url_raw( wp_get_referer() ) ); + exit(); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packaging/ReportQueue.php b/packages/woocommerce-germanized-shipments/src/Packaging/ReportQueue.php new file mode 100644 index 000000000..047953647 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packaging/ReportQueue.php @@ -0,0 +1,337 @@ +diff( $args['end'] ); + + // Add version + $args['version'] = Package::get_version(); + + $generator = new AsyncReportGenerator( $type, $args ); + $queue_args = $generator->get_args(); + $queue = self::get_queue(); + + self::cancel( $generator->get_id() ); + + $report = $generator->start(); + + if ( is_a( $report, '\Vendidero\Germanized\Shipments\Packaging\Report' ) && $report->exists() ) { + Package::log( sprintf( 'Starting new %1$s', $report->get_title() ) ); + Package::log( sprintf( 'Default report arguments: %s', wc_print_r( $queue_args, true ) ) ); + + $queue->schedule_single( + time() + 10, + self::get_hook_name( $generator->get_id() ), + array( 'args' => $queue_args ), + 'woocommerce_gzd_shipments' + ); + + $running = self::get_reports_running(); + + if ( ! in_array( $generator->get_id(), $running, true ) ) { + $running[] = $generator->get_id(); + } + + update_option( 'woocommerce_gzd_shipments_packaging_reports_running', $running, false ); + self::clear_cache(); + + return $generator->get_id(); + } + + return false; + } + + public static function clear_cache() { + wp_cache_delete( 'woocommerce_gzd_shipments_packaging_reports_running', 'options' ); + } + + public static function get_queue_details( $report_id ) { + $details = array( + 'next_date' => null, + 'link' => admin_url( 'admin.php?page=wc-status&tab=action-scheduler&s=' . esc_attr( $report_id ) . '&status=pending' ), + 'shipment_count' => 0, + 'has_action' => false, + 'is_finished' => false, + 'action' => false, + ); + + if ( $queue = self::get_queue() ) { + if ( $next_date = $queue->get_next( self::get_hook_name( $report_id ) ) ) { + $details['next_date'] = $next_date; + } + + $search_args = array( + 'hook' => self::get_hook_name( $report_id ), + 'status' => \ActionScheduler_Store::STATUS_RUNNING, + 'order' => 'DESC', + 'per_page' => 1, + ); + + $results = $queue->search( $search_args ); + + /** + * Search for pending as fallback + */ + if ( empty( $results ) ) { + $search_args['status'] = \ActionScheduler_Store::STATUS_PENDING; + $results = $queue->search( $search_args ); + } + + /** + * Last resort: Search for completed (e.g. if no pending and no running are found - must have been completed) + */ + if ( empty( $results ) ) { + $search_args['status'] = \ActionScheduler_Store::STATUS_COMPLETE; + $results = $queue->search( $search_args ); + } + + if ( ! empty( $results ) ) { + $action = array_values( $results )[0]; + $args = $action->get_args(); + $processed = isset( $args['args']['processed'] ) ? (int) $args['args']['processed'] : 0; + + $details['shipment_count'] = absint( $processed ); + $details['has_action'] = true; + $details['action'] = $action; + $details['is_finished'] = $action->is_finished(); + } + } + + return $details; + } + + public static function get_batch_size() { + return apply_filters( 'woocommerce_gzd_shipments_packaging_report_batch_size', 25 ); + } + + public static function get_shipment_statuses() { + $statuses = array_keys( wc_gzd_get_shipment_statuses() ); + $statuses = array_diff( $statuses, array( 'gzd-draft', 'gzd-requested' ) ); + + return apply_filters( 'woocommerce_gzd_shipments_packaging_report_valid_statuses', $statuses ); + } + + /** + * @param $args + * + * @return \Vendidero\Germanized\Shipments\Shipment[] + */ + public static function query( $args ) { + $query_args = array( + 'date_created' => $args['start'] . '...' . $args['end'], + 'offset' => $args['offset'], + 'type' => $args['type'], + 'status' => $args['status'], + 'limit' => $args['limit'], + ); + + return wc_gzd_get_shipments( $query_args ); + } + + public static function cancel( $id ) { + $data = ReportHelper::get_report_data( $id ); + $generator = new AsyncReportGenerator( $data['type'], $data ); + $queue = self::get_queue(); + $running = self::get_reports_running(); + + if ( self::is_running( $id ) ) { + $running = array_diff( $running, array( $id ) ); + Package::log( sprintf( 'Cancelled %s', ReportHelper::get_report_title( $id ) ) ); + + update_option( 'woocommerce_gzd_shipments_packaging_reports_running', $running, false ); + self::clear_cache(); + $generator->delete(); + } + + /** + * Cancel outstanding events and queue new. + */ + $queue->cancel_all( self::get_hook_name( $id ) ); + } + + public static function get_queue() { + return function_exists( 'WC' ) ? WC()->queue() : false; + } + + public static function is_running( $id ) { + $running = self::get_reports_running(); + + if ( in_array( $id, $running, true ) && self::get_queue()->get_next( self::get_hook_name( $id ) ) ) { + return true; + } + + return false; + } + + public static function get_hook_name( $id ) { + if ( ! strstr( $id, 'woocommerce_gzd_shipments_' ) ) { + $id = 'woocommerce_gzd_shipments_' . $id; + } + + return $id; + } + + public static function next( $type, $args ) { + $generator = new AsyncReportGenerator( $type, $args ); + $result = $generator->next(); + $is_empty = false; + $queue = self::get_queue(); + + if ( is_wp_error( $result ) ) { + $is_empty = $result->get_error_message( 'empty' ); + } + + if ( ! $is_empty ) { + $new_args = $generator->get_args(); + + // Increase offset + $new_args['offset'] = (int) $new_args['offset'] + (int) $new_args['limit']; + + $queue->cancel_all( self::get_hook_name( $generator->get_id() ) ); + + Package::log( sprintf( 'Starting new queue: %s', wc_print_r( $new_args, true ) ) ); + + $queue->schedule_single( + time() + 10, + self::get_hook_name( $generator->get_id() ), + array( 'args' => $new_args ), + 'woocommerce_gzd_shipments' + ); + } else { + self::complete( $generator ); + } + } + + /** + * @param AsyncReportGenerator $generator + */ + public static function complete( $generator ) { + $queue = self::get_queue(); + $type = $generator->get_type(); + + /** + * Cancel outstanding events. + */ + $queue->cancel_all( self::get_hook_name( $generator->get_id() ) ); + + $report = $generator->complete(); + $status = 'failed'; + + if ( is_a( $report, '\Vendidero\Germanized\Shipments\Packaging\Report' ) && $report->exists() ) { + $status = 'completed'; + } + + Package::log( sprintf( 'Completed %1$s. Status: %2$s', $report->get_title(), $status ) ); + + self::maybe_stop_report( $report->get_id() ); + } + + public static function maybe_stop_report( $report_id ) { + $reports_running = self::get_reports_running(); + + if ( in_array( $report_id, $reports_running, true ) ) { + $reports_running = array_diff( $reports_running, array( $report_id ) ); + update_option( 'woocommerce_gzd_shipments_packaging_reports_running', $reports_running, false ); + + if ( $queue = self::get_queue() ) { + $queue->cancel_all( self::get_hook_name( $report_id ) ); + } + + /** + * Force non-cached running option + */ + wp_cache_delete( 'woocommerce_gzd_shipments_packaging_reports_running', 'options' ); + + return true; + } + + return false; + } + + public static function get_reports_running() { + return (array) get_option( 'woocommerce_gzd_shipments_packaging_reports_running', array() ); + } + + public static function get_timeframe( $type, $date = null, $date_end = null ) { + $date_start = null; + $date_end = is_null( $date_end ) ? null : $date_end; + $start_indicator = is_null( $date ) ? new \WC_DateTime() : $date; + + if ( ! is_a( $start_indicator, 'WC_DateTime' ) && is_numeric( $start_indicator ) ) { + $start_indicator = new \WC_DateTime( '@' . $start_indicator ); + } + + if ( ! is_null( $date_end ) && ! is_a( $date_end, 'WC_DateTime' ) && is_numeric( $date_end ) ) { + $date_end = new \WC_DateTime( '@' . $date_end ); + } + + if ( 'quarterly' === $type ) { + $month = $start_indicator->date( 'n' ); + $quarter = (int) ceil( $month / 3 ); + $start_month = 'Jan'; + $end_month = 'Mar'; + + if ( 2 === $quarter ) { + $start_month = 'Apr'; + $end_month = 'Jun'; + } elseif ( 3 === $quarter ) { + $start_month = 'Jul'; + $end_month = 'Sep'; + } elseif ( 4 === $quarter ) { + $start_month = 'Oct'; + $end_month = 'Dec'; + } + + $date_start = new \WC_DateTime( 'first day of ' . $start_month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + $date_end = new \WC_DateTime( 'last day of ' . $end_month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + } elseif ( 'monthly' === $type ) { + $month = $start_indicator->format( 'M' ); + + $date_start = new \WC_DateTime( 'first day of ' . $month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + $date_end = new \WC_DateTime( 'last day of ' . $month . ' ' . $start_indicator->format( 'Y' ) . ' midnight' ); + } elseif ( 'yearly' === $type ) { + $date_end = clone $start_indicator; + $date_start = clone $start_indicator; + + $date_end->modify( 'last day of dec ' . $start_indicator->format( 'Y' ) . ' midnight' ); + $date_start->modify( 'first day of jan ' . $start_indicator->format( 'Y' ) . ' midnight' ); + } else { + if ( is_null( $date_end ) ) { + $date_end = clone $start_indicator; + $date_end->modify( '-1 year' ); + } + + $date_start = clone $start_indicator; + } + + /** + * Always set start and end time to midnight + */ + if ( $date_start ) { + $date_start->setTime( 0, 0 ); + } + + if ( $date_end ) { + $date_end->setTime( 0, 0 ); + } + + return array( + 'start' => $date_start, + 'end' => $date_end, + ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/PackagingFactory.php b/packages/woocommerce-germanized-shipments/src/PackagingFactory.php new file mode 100644 index 000000000..4c48f8ec3 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/PackagingFactory.php @@ -0,0 +1,65 @@ +get_id(); + } elseif ( ! empty( $packaging->packaging_id ) ) { + return $packaging->packaging_id; + } else { + return false; + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packing/Helper.php b/packages/woocommerce-germanized-shipments/src/Packing/Helper.php new file mode 100644 index 000000000..24c0abbb2 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packing/Helper.php @@ -0,0 +1,31 @@ +get_id() ] = new PackagingBox( $packaging ); + } + } + + if ( $id ) { + return array_key_exists( $id, self::$packaging ) ? self::$packaging[ $id ] : false; + } + + return self::$packaging; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packing/OrderItem.php b/packages/woocommerce-germanized-shipments/src/Packing/OrderItem.php new file mode 100644 index 000000000..1dc130ee0 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packing/OrderItem.php @@ -0,0 +1,117 @@ +item = $item; + + if ( ! is_callable( array( $item, 'get_product' ) ) ) { + throw new \Exception( 'Invalid item' ); + } + + if ( $product = $this->item->get_product() ) { + $this->product = $product; + + $width = empty( $this->product->get_width() ) ? 0 : wc_format_decimal( $this->product->get_width() ); + $length = empty( $this->product->get_length() ) ? 0 : wc_format_decimal( $this->product->get_length() ); + $depth = empty( $this->product->get_height() ) ? 0 : wc_format_decimal( $this->product->get_height() ); + + $this->dimensions = array( + 'width' => (int) wc_get_dimension( $width, 'mm' ), + 'length' => (int) wc_get_dimension( $length, 'mm' ), + 'depth' => (int) wc_get_dimension( $depth, 'mm' ), + ); + + $weight = empty( $this->product->get_weight() ) ? 0 : wc_format_decimal( $this->product->get_weight() ); + $this->weight = (int) wc_get_weight( $weight, 'g' ); + } + + if ( ! $product ) { + throw new \Exception( 'Missing product' ); + } + } + + public function get_id() { + return $this->item->get_id(); + } + + /** + * @return \WC_Order_Item_Product + */ + public function get_order_item() { + return $this->item; + } + + /** + * Item SKU etc. + */ + public function getDescription(): string { + if ( $this->product->get_sku() ) { + return $this->product->get_sku(); + } + + return $this->item->get_id(); + } + + /** + * Item width in mm. + */ + public function getWidth(): int { + return $this->dimensions['width']; + } + + /** + * Item length in mm. + */ + public function getLength(): int { + return $this->dimensions['length']; + } + + /** + * Item depth in mm. + */ + public function getDepth(): int { + return $this->dimensions['depth']; + } + + /** + * Item weight in g. + */ + public function getWeight(): int { + return $this->weight; + } + + /** + * Does this item need to be kept flat / packed "this way up"? + */ + public function getKeepFlat(): bool { + return false; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packing/PackagingBox.php b/packages/woocommerce-germanized-shipments/src/Packing/PackagingBox.php new file mode 100644 index 000000000..6abe9c7bb --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packing/PackagingBox.php @@ -0,0 +1,147 @@ +packaging = $packaging; + + $width = empty( $this->packaging->get_width() ) ? 0 : wc_format_decimal( $this->packaging->get_width() ); + $length = empty( $this->packaging->get_length() ) ? 0 : wc_format_decimal( $this->packaging->get_length() ); + $depth = empty( $this->packaging->get_height() ) ? 0 : wc_format_decimal( $this->packaging->get_height() ); + + $this->dimensions = array( + 'width' => (int) floor( wc_get_dimension( $width, 'mm', wc_gzd_get_packaging_dimension_unit() ) ), + 'length' => (int) floor( wc_get_dimension( $length, 'mm', wc_gzd_get_packaging_dimension_unit() ) ), + 'depth' => (int) floor( wc_get_dimension( $depth, 'mm', wc_gzd_get_packaging_dimension_unit() ) ), + ); + + $weight = empty( $this->packaging->get_weight() ) ? 0 : wc_format_decimal( $this->packaging->get_weight() ); + $this->weight = (int) floor( wc_get_weight( $weight, 'g', wc_gzd_get_packaging_weight_unit() ) ); + + $max_content_weight = empty( $this->packaging->get_max_content_weight() ) ? 0 : wc_format_decimal( $this->packaging->get_max_content_weight() ); + $this->max_weight = (int) floor( wc_get_weight( $max_content_weight, 'g', wc_gzd_get_packaging_weight_unit() ) ); + + /** + * If no max weight was chosen - use 50kg as fallback + */ + if ( empty( $this->max_weight ) ) { + $this->max_weight = 50000; + } + } + + public function get_id() { + return $this->packaging->get_id(); + } + + /** + * Reference for box type (e.g. SKU or description). + */ + public function getReference(): string { + return (string) $this->packaging->get_title(); + } + + /** + * Outer width in mm. + */ + public function getOuterWidth(): int { + return $this->dimensions['width']; + } + + /** + * Outer length in mm. + */ + public function getOuterLength(): int { + return $this->dimensions['length']; + } + + /** + * Outer depth in mm. + */ + public function getOuterDepth(): int { + return $this->dimensions['depth']; + } + + /** + * Empty weight in g. + */ + public function getEmptyWeight(): int { + return $this->weight; + } + + /** + * Returns the threshold by which the inner dimension gets reduced + * in comparison to the outer dimension. + * + * @param string $type + * + * @return float + */ + public function get_inner_dimension_buffer( $value, $type = 'width' ) { + if ( apply_filters( 'woocommerce_gzd_packaging_inner_dimension_use_percentage_buffer', false, $type, $this ) ) { + $percentage_buffer = apply_filters( 'woocommerce_gzd_packaging_inner_dimension_percentage_buffer', 0.5, $type, $this ) / 100; + $value = $value - ( $value * $percentage_buffer ); + } else { + $fixed_buffer = apply_filters( 'woocommerce_gzd_packaging_inner_dimension_fixed_buffer_mm', 5, $type, $this ); + $value = $value - $fixed_buffer; + } + + return max( $value, 0 ); + } + + /** + * Inner width in mm. + */ + public function getInnerWidth(): int { + $width = $this->get_inner_dimension_buffer( $this->dimensions['width'], 'width' ); + + return $width; + } + + /** + * Inner length in mm. + */ + public function getInnerLength(): int { + $length = $this->get_inner_dimension_buffer( $this->dimensions['length'], 'length' ); + + return $length; + } + + /** + * Inner depth in mm. + */ + public function getInnerDepth(): int { + $depth = $this->get_inner_dimension_buffer( $this->dimensions['depth'], 'depth' ); + + return $depth; + } + + /** + * Max weight the packaging can hold in g. + */ + public function getMaxWeight(): int { + return $this->max_weight; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Packing/ShipmentItem.php b/packages/woocommerce-germanized-shipments/src/Packing/ShipmentItem.php new file mode 100644 index 000000000..e4a347676 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Packing/ShipmentItem.php @@ -0,0 +1,111 @@ +item = $item; + + if ( $shipment = $item->get_shipment() ) { + $dimension_unit = $shipment->get_dimension_unit(); + $weight_unit = $shipment->get_weight_unit(); + } else { + $dimension_unit = get_option( 'woocommerce_dimension_unit', 'cm' ); + $weight_unit = get_option( 'woocommerce_weight_unit', 'kg' ); + } + + $width = empty( $this->item->get_width() ) ? 0 : wc_format_decimal( $this->item->get_width() ); + $length = empty( $this->item->get_length() ) ? 0 : wc_format_decimal( $this->item->get_length() ); + $depth = empty( $this->item->get_height() ) ? 0 : wc_format_decimal( $this->item->get_height() ); + + $this->dimensions = array( + 'width' => (int) ceil( wc_get_dimension( $width, 'mm', $dimension_unit ) ), + 'length' => (int) ceil( wc_get_dimension( $length, 'mm', $dimension_unit ) ), + 'depth' => (int) ceil( wc_get_dimension( $depth, 'mm', $dimension_unit ) ), + ); + + $weight = empty( $this->item->get_weight() ) ? 0 : wc_format_decimal( $this->item->get_weight() ); + $this->weight = (int) ceil( wc_get_weight( $weight, 'g', $weight_unit ) ); + } + + /** + * @return \Vendidero\Germanized\Shipments\ShipmentItem + */ + public function get_shipment_item() { + return $this->item; + } + + public function get_id() { + return $this->item->get_id(); + } + + /** + * Item SKU etc. + */ + public function getDescription(): string { + if ( $this->item->get_sku() ) { + return $this->item->get_sku(); + } + + return $this->item->get_id(); + } + + /** + * Item width in mm. + */ + public function getWidth(): int { + return $this->dimensions['width']; + } + + /** + * Item length in mm. + */ + public function getLength(): int { + return $this->dimensions['length']; + } + + /** + * Item depth in mm. + */ + public function getDepth(): int { + return $this->dimensions['depth']; + } + + /** + * Item weight in g. + */ + public function getWeight(): int { + return $this->weight; + } + + /** + * Does this item need to be kept flat / packed "this way up"? + */ + public function getKeepFlat(): bool { + return false; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Product.php b/packages/woocommerce-germanized-shipments/src/Product.php new file mode 100644 index 000000000..d1e739d1c --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Product.php @@ -0,0 +1,130 @@ +product = $product; + } + + /** + * Returns the Woo WC_Product original object + * + * @return object|WC_Product + */ + public function get_product() { + return $this->product; + } + + protected function get_forced_parent_product() { + if ( $this->product->is_type( 'variation' ) ) { + if ( $parent = wc_get_product( $this->product->get_parent_id() ) ) { + return $parent; + } + } + + return $this->product; + } + + public function get_hs_code( $context = 'view' ) { + $legacy_data = $this->get_forced_parent_product()->get_meta( '_dhl_hs_code', true, $context ); + $data = $this->get_forced_parent_product()->get_meta( '_hs_code', true, $context ); + + if ( '' === $data && ! empty( $legacy_data ) ) { + $data = $legacy_data; + } + + return $data; + } + + public function get_manufacture_country( $context = 'view' ) { + $legacy_data = $this->get_forced_parent_product()->get_meta( '_dhl_manufacture_country', true, $context ); + $data = $this->get_forced_parent_product()->get_meta( '_manufacture_country', true, $context ); + + if ( '' === $data && ! empty( $legacy_data ) ) { + $data = $legacy_data; + } + + if ( '' === $data && 'view' === $context ) { + return wc_get_base_location()['country']; + } + + return $data; + } + + public function get_main_category() { + $ids = $this->get_forced_parent_product()->get_category_ids(); + $term_name = ''; + + if ( ! empty( $ids ) ) { + foreach ( $ids as $term_id ) { + $term = get_term( $term_id, 'product_cat' ); + + if ( empty( $term->slug ) ) { + continue; + } + + $term_name = $term->name; + break; + } + } + + return $term_name; + } + + public function set_hs_code( $code ) { + $this->product->update_meta_data( '_hs_code', $code ); + } + + public function set_manufacture_country( $country ) { + $this->product->update_meta_data( '_manufacture_country', substr( wc_strtoupper( $country ), 0, 2 ) ); + } + + /** + * Call child methods if the method does not exist. + * + * @param $method + * @param $args + * + * @return bool|mixed + */ + public function __call( $method, $args ) { + if ( method_exists( $this->product, $method ) ) { + return call_user_func_array( array( $this->product, $method ), $args ); + } + + return false; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Rest/ShipmentsController.php b/packages/woocommerce-germanized-shipments/src/Rest/ShipmentsController.php new file mode 100644 index 000000000..d183cd932 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Rest/ShipmentsController.php @@ -0,0 +1,2057 @@ +namespace, + '/' . $this->rest_base, + array( + array( + 'methods' => WP_REST_Server::READABLE, + 'callback' => array( $this, 'get_items' ), + 'permission_callback' => array( $this, 'get_items_permissions_check' ), + 'args' => $this->get_collection_params(), + ), + array( + 'methods' => WP_REST_Server::CREATABLE, + 'callback' => array( $this, 'create_item' ), + 'permission_callback' => array( $this, 'create_item_permissions_check' ), + 'args' => $this->get_endpoint_args_for_item_schema( WP_REST_Server::CREATABLE ), + ), + 'schema' => array( $this, 'get_public_item_schema' ), + ) + ); + register_rest_route( + $this->namespace, + '/' . $this->rest_base . '/(?P[\d]+)', + array( + array( + 'methods' => WP_REST_Server::READABLE, + 'callback' => array( $this, 'get_item' ), + 'permission_callback' => array( $this, 'get_item_permissions_check' ), + 'args' => array( + 'context' => $this->get_context_param( array( 'default' => 'view' ) ), + ), + ), + array( + 'methods' => WP_REST_Server::EDITABLE, + 'callback' => array( $this, 'update_item' ), + 'permission_callback' => array( $this, 'update_item_permissions_check' ), + 'args' => $this->get_endpoint_args_for_item_schema( WP_REST_Server::EDITABLE ), + ), + array( + 'methods' => WP_REST_Server::DELETABLE, + 'callback' => array( $this, 'delete_item' ), + 'permission_callback' => array( $this, 'delete_item_permissions_check' ), + 'args' => array( + 'force' => array( + 'default' => false, + 'type' => 'boolean', + 'description' => _x( 'Whether to bypass trash and force deletion.', 'shipments', 'woocommerce-germanized' ), + ), + ), + ), + 'schema' => array( $this, 'get_public_item_schema' ), + ) + ); + + register_rest_route( + $this->namespace, + '/' . $this->rest_base . '/(?P[\d]+)/label', + array( + array( + 'methods' => WP_REST_Server::READABLE, + 'callback' => array( $this, 'get_label' ), + 'permission_callback' => array( $this, 'get_label_permissions_check' ), + 'args' => array( + 'context' => $this->get_context_param( array( 'default' => 'view' ) ), + ), + ), + array( + 'methods' => WP_REST_Server::CREATABLE, + 'callback' => array( $this, 'create_label' ), + 'permission_callback' => array( $this, 'create_label_permissions_check' ), + 'args' => array( + array( + 'description' => _x( 'Shipment label.', 'shipment', 'woocommerce-germanized' ), + 'context' => array( 'view', 'edit' ), + 'readonly' => false, + 'type' => 'object', + 'properties' => array( + 'type' => 'object', + 'properties' => array( + 'key' => array( + 'description' => _x( 'Label field key.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'value' => array( + 'description' => _x( 'Label field value.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'mixed', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + ), + ), + ), + array( + 'methods' => WP_REST_Server::DELETABLE, + 'callback' => array( $this, 'delete_label' ), + 'permission_callback' => array( $this, 'delete_label_permissions_check' ), + 'args' => array( + 'force' => array( + 'default' => false, + 'type' => 'boolean', + 'description' => _x( 'Whether to bypass trash and force deletion.', 'shipments', 'woocommerce-germanized' ), + ), + ), + ), + 'schema' => array( $this, 'get_public_item_label_schema' ), + ) + ); + } + + /** + * Get object. + * + * @param int|Shipment $id Object ID. + * @return Shipment Shipment object or WP_Error object. + */ + protected function get_object( $id ) { + return $this->get_shipment( $id ); + } + + /** + * Get object permalink. + * + * @param Shipment $shipment Object. + * @return string + */ + protected function get_permalink( $shipment ) { + return $shipment->get_edit_shipment_url(); + } + + private static function get_shipment_statuses() { + return array_map( array( 'Vendidero\Germanized\Shipments\Api', 'remove_status_prefix' ), array_keys( wc_gzd_get_shipment_statuses() ) ); + } + + /** + * Checks if a given request has access to get a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has read access for the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function get_item_permissions_check( $request ) { + if ( ! $this->check_permissions( 'shipment', 'read', $request['id'] ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_view', _x( 'Sorry, you are not allowed to view this resource.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + /** + * Retrieves a shipment by id. + * + * @param int $shipment_id + * + * @return Shipment|false + */ + private function get_shipment( $shipment_id ) { + $shipment = wc_gzd_get_shipment( $shipment_id ); + + return $shipment; + } + + /** + * Checks if a given request has access to get a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has read access for the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function get_items_permissions_check( $request ) { + if ( ! $this->check_permissions() ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_view', _x( 'Sorry, you cannot list resources.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + protected function check_permissions( $object_type = 'shipment', $context = 'read', $object_id = 0 ) { + if ( 'delete' === $context || 'edit' === $context ) { + $post_type_object = get_post_type_object( 'shop_order' ); + $capped = 'delete' === $context ? $post_type_object->cap->delete_posts : $post_type_object->cap->edit_posts; + $permission = current_user_can( $capped, $object_id ); + } else { + $permission = wc_rest_check_post_permissions( 'shop_order', $context ); + } + + return apply_filters( 'woocommerce_gzd_shipments_rest_check_permissions', $permission, $object_type, $context, $object_id ); + } + + /** + * Retrieves a collection of items. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure. + * @since 4.7.0 + */ + public function get_items( $request ) { + $prepared_args = array( + 'limit' => $request['per_page'], + 'paginate' => true, + 'type' => $request['type'], + 'order_id' => $request['order_id'], + 'search' => $request['search'], + 'status' => $request['status'], + 'order' => $request['order'], + 'orderby' => $request['orderby'], + 'count_total' => true, + ); + + if ( ! empty( $prepared_args['search'] ) ) { + $prepared_args['search'] = '*' . $prepared_args['search'] . '*'; + } + + if ( ! empty( $request['offset'] ) ) { + $prepared_args['offset'] = $request['offset']; + } else { + $prepared_args['offset'] = ( $request['page'] - 1 ) * $prepared_args['limit']; + } + + $objects = array(); + $query = new \Vendidero\Germanized\Shipments\ShipmentQuery( $prepared_args ); + $shipments = $query->get_shipments(); + + if ( ! empty( $shipments ) ) { + foreach ( $shipments as $shipment ) { + if ( ! $this->check_permissions( 'shipment', 'read', $shipment->get_id() ) ) { + continue; + } + + $objects[] = $this->prepare_object_for_response( $shipment, $request ); + } + } + + $page = (int) $request['page']; + $max_pages = $query->get_max_num_pages(); + + $response = rest_ensure_response( $objects ); + $response->header( 'X-WP-Total', $query->get_total() ); + $response->header( 'X-WP-TotalPages', (int) $max_pages ); + + $base = $this->rest_base; + $attrib_prefix = '(?P<'; + if ( strpos( $base, $attrib_prefix ) !== false ) { + $attrib_names = array(); + preg_match( '/\(\?P<[^>]+>.*\)/', $base, $attrib_names, PREG_OFFSET_CAPTURE ); + foreach ( $attrib_names as $attrib_name_match ) { + $beginning_offset = strlen( $attrib_prefix ); + $attrib_name_end = strpos( $attrib_name_match[0], '>', $attrib_name_match[1] ); + $attrib_name = substr( $attrib_name_match[0], $beginning_offset, $attrib_name_end - $beginning_offset ); + if ( isset( $request[ $attrib_name ] ) ) { + $base = str_replace( "(?P<$attrib_name>[\d]+)", $request[ $attrib_name ], $base ); + } + } + } + $base = add_query_arg( $request->get_query_params(), rest_url( sprintf( '/%s/%s', $this->namespace, $base ) ) ); + + if ( $page > 1 ) { + $prev_page = $page - 1; + if ( $prev_page > $max_pages ) { + $prev_page = $max_pages; + } + $prev_link = add_query_arg( 'page', $prev_page, $base ); + $response->link_header( 'prev', $prev_link ); + } + if ( $max_pages > $page ) { + $next_page = $page + 1; + $next_link = add_query_arg( 'page', $next_page, $base ); + $response->link_header( 'next', $next_link ); + } + + return $response; + } + + /** + * Checks if a given request has access to update a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has access to update the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function update_item_permissions_check( $request ) { + if ( ! $this->check_permissions( 'shipment', 'edit', $request['id'] ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_edit', _x( 'Sorry, you are not allowed to edit this resource.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + /** + * Checks if a given request has access to create a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has access to create the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function create_item_permissions_check( $request ) { + if ( ! $this->check_permissions( 'shipment', 'create' ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_view', _x( 'Sorry, you are not allowed to create resources.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + /** + * @param $request + * @param boolean $creating + * + * @return Shipment + * @throws \WC_REST_Exception + */ + protected function prepare_object_for_database( $request, $creating = false ) { + $id = isset( $request['id'] ) ? absint( $request['id'] ) : false; + + // Type is the most important part here because we need to be using the correct class and methods. + if ( isset( $request['type'] ) ) { + $shipment = ShipmentFactory::get_shipment( $id, $request['type'] ); + } elseif ( $id ) { + $shipment = wc_gzd_get_shipment( $id ); + } else { + $shipment = ShipmentFactory::get_shipment( false ); + } + + if ( ! $shipment ) { + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_id', _x( 'There was an error while creating the shipment.', 'shipments', 'woocommerce-germanized' ) ); + } + + if ( isset( $request['order_id'] ) ) { + $shipment->set_order_id( absint( wp_unslash( $request['order_id'] ) ) ); + } + + if ( $creating ) { + $order_shipment = $shipment->get_order_shipment(); + + if ( ! $order_shipment ) { + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_id', _x( 'This order does not exist.', 'shipments', 'woocommerce-germanized' ) ); + } + + if ( 'return' === $shipment->get_type() ) { + if ( ! $order_shipment->needs_return() ) { + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_id', _x( 'This order does need a return.', 'shipments', 'woocommerce-germanized' ) ); + } + } else { + if ( ! $order_shipment->needs_shipping() ) { + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_id', _x( 'This order does need shipping.', 'shipments', 'woocommerce-germanized' ) ); + } + } + + $shipment->sync(); + } + + if ( isset( $request['shipping_provider'] ) ) { + $provider = wc_clean( wp_unslash( $request['shipping_provider'] ) ); + + if ( $provider = wc_gzd_get_shipping_provider( $provider ) ) { + $shipment->set_shipping_provider( $provider ); + } + } + + if ( isset( $request['shipping_method'] ) ) { + $shipment->set_shipping_method( wc_clean( wp_unslash( $request['shipping_method'] ) ) ); + } + + if ( isset( $request['packaging_id'] ) ) { + $packaging_id = absint( wp_unslash( $request['packaging_id'] ) ); + + if ( $packaging = wc_gzd_get_packaging( $packaging_id ) ) { + $shipment->set_packaging_id( $packaging_id ); + } + } + + if ( isset( $request['packaging_weight'] ) ) { + $shipment->set_packaging_weight( wc_clean( wp_unslash( $request['packaging_weight'] ) ) ); + } + + if ( isset( $request['tracking_id'] ) ) { + $shipment->set_tracking_id( wc_clean( wp_unslash( $request['tracking_id'] ) ) ); + } + + if ( isset( $request['dimensions'] ) ) { + if ( isset( $request['dimensions']['length'] ) ) { + $shipment->set_length( wc_clean( wp_unslash( $request['dimensions']['length'] ) ) ); + } + if ( isset( $request['dimensions']['width'] ) ) { + $shipment->set_width( wc_clean( wp_unslash( $request['dimensions']['width'] ) ) ); + } + if ( isset( $request['dimensions']['height'] ) ) { + $shipment->set_height( wc_clean( wp_unslash( $request['dimensions']['height'] ) ) ); + } + } + + if ( isset( $request['dimension_unit'] ) ) { + $shipment->set_dimension_unit( wc_clean( wp_unslash( $request['dimension_unit'] ) ) ); + } + + if ( isset( $request['weight'] ) ) { + $shipment->set_weight( wc_clean( wp_unslash( $request['weight'] ) ) ); + } + + if ( isset( $request['weight_unit'] ) ) { + $shipment->set_weight_unit( wc_clean( wp_unslash( $request['weight_unit'] ) ) ); + } + + if ( isset( $request['address'] ) && is_array( $request['address'] ) ) { + $shipment->set_address( wc_clean( wp_unslash( $request['address'] ) ) ); + + if ( isset( $request['address']['country'] ) ) { + $shipment->set_country( wc_clean( wp_unslash( $request['address']['country'] ) ) ); + } + } + + if ( isset( $request['total'] ) ) { + $shipment->set_total( wc_clean( wp_unslash( $request['total'] ) ) ); + } + + if ( isset( $request['subtotal'] ) ) { + $shipment->set_subtotal( wc_clean( wp_unslash( $request['subtotal'] ) ) ); + } + + if ( isset( $request['additional_total'] ) ) { + $shipment->set_additional_total( wc_clean( wp_unslash( $request['additional_total'] ) ) ); + } + + if ( is_a( $shipment, 'Vendidero\Germanized\Shipments\ReturnShipment' ) ) { + if ( isset( $request['sender_address'] ) && is_array( $request['sender_address'] ) ) { + $shipment->set_sender_address( wc_clean( wp_unslash( $request['sender_address'] ) ) ); + } + + if ( isset( $request['is_customer_requested'] ) ) { + $shipment->set_is_customer_requested( wc_clean( wp_unslash( $request['is_customer_requested'] ) ) ); + } + } + + if ( ! empty( $request['date_created'] ) ) { + $date = rest_parse_date( wc_clean( wp_unslash( $request['date_created'] ) ) ); + + if ( $date ) { + $shipment->set_date_created( $date ); + } + } + + if ( ! empty( $request['date_created_gmt'] ) ) { + $date = rest_parse_date( wc_clean( wp_unslash( $request['date_created_gmt'] ) ), true ); + + if ( $date ) { + $shipment->set_date_created( $date ); + } + } + + if ( ! empty( $request['est_delivery_date'] ) ) { + $date = rest_parse_date( wc_clean( wp_unslash( $request['est_delivery_date'] ) ) ); + + if ( $date ) { + $shipment->set_est_delivery_date( $date ); + } + } + + if ( ! empty( $request['est_delivery_date_gmt'] ) ) { + $date = rest_parse_date( wc_clean( wp_unslash( $request['est_delivery_date_gmt'] ) ), true ); + + if ( $date ) { + $shipment->set_est_delivery_date( $date ); + } + } + + if ( ! empty( $request['date_sent'] ) ) { + $date = rest_parse_date( wc_clean( wp_unslash( $request['date_sent'] ) ) ); + + if ( $date ) { + $shipment->set_date_sent( $date ); + } + } + + if ( ! empty( $request['date_sent_gmt'] ) ) { + $date = rest_parse_date( wc_clean( wp_unslash( $request['date_sent_gmt'] ) ), true ); + + if ( $date ) { + $shipment->set_date_sent( $date ); + } + } + + if ( ! empty( $request['items'] ) && is_array( $request['items'] ) ) { + foreach ( $request['items'] as $item ) { + if ( is_array( $item ) ) { + if ( $this->item_is_null( $item ) || ( isset( $item['quantity'] ) && 0 === $item['quantity'] ) ) { + $shipment->remove_item( $item['id'] ); + } else { + $this->set_item( $shipment, $item ); + } + } + } + } elseif ( $creating ) { + $shipment->sync_items(); + } + + // Update the status at last + if ( isset( $request['status'] ) ) { + $status = str_replace( 'gzd-', '', wc_clean( wp_unslash( $request['status'] ) ) ); + + if ( in_array( $status, self::get_shipment_statuses(), true ) ) { + $shipment->set_status( $status ); + } + } + + if ( isset( $request['meta_data'] ) && is_array( $request['meta_data'] ) ) { + foreach ( $request['meta_data'] as $meta ) { + $meta = wc_clean( wp_unslash( $meta ) ); + + if ( isset( $meta['key'] ) ) { + $value = isset( $meta['value'] ) ? $meta['value'] : null; + $shipment->update_meta_data( $meta['key'], $value, isset( $meta['id'] ) ? $meta['id'] : '' ); + } + } + } + + if ( $shipment->get_item_count() <= 0 ) { + $shipment->delete( true ); + + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_id', _x( 'This shipment does not contain any items and was deleted.', 'shipments', 'woocommerce-germanized' ) ); + } + + /** + * Filters a shipment before it is inserted via the REST API. + * + * @param Shipment $shipment Shipment object. + * @param WP_REST_Request $request Request object. + */ + return apply_filters( 'woocommerce_gzd_rest_pre_insert_shipment_object', $shipment, $request ); + } + + /** + * Wrapper method to create/update order items. + * When updating, the item ID provided is checked to ensure it is associated + * with the order. + * + * @param Shipment $shipment order object. + * @param array $posted item provided in the request body. + * + * @throws \WC_REST_Exception If item ID is not associated with order. + */ + protected function set_item( $shipment, $posted ) { + if ( ! empty( $posted['id'] ) ) { + $action = 'update'; + } else { + $action = 'create'; + } + + $item = null; + + // Verify provided line item ID is associated with order. + if ( 'update' === $action ) { + $item = $shipment->get_item( absint( $posted['id'] ) ); + + if ( ! $item ) { + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_item_id', _x( 'Shipment item ID provided is not associated with shipment.', 'shipments', 'woocommerce-germanized' ), 400 ); + } + } + + if ( is_null( $item ) ) { + if ( 'return' === $shipment->get_type() ) { + $item = new \Vendidero\Germanized\Shipments\ShipmentReturnItem(); + } else { + $item = new \Vendidero\Germanized\Shipments\ShipmentItem(); + } + } + + if ( isset( $posted['order_item_id'] ) ) { + $item->set_order_item_id( absint( wp_unslash( $posted['order_item_id'] ) ) ); + } + + $item->set_shipment( $shipment ); + + /** + * Sync quantity first. + */ + if ( 'create' === $action ) { + $quantity = isset( $posted['quantity'] ) ? absint( wp_unslash( $posted['quantity'] ) ) : -1; + $quantity_left = 0; + + if ( $order_shipment = $shipment->get_order_shipment() ) { + if ( 'return' === $shipment->get_type() ) { + $quantity_left = $order_shipment->get_item_quantity_left_for_returning( $item->get_order_item_id() ); + } elseif ( $order_item = $item->get_order_item() ) { + $quantity_left = $order_shipment->get_item_quantity_left_for_shipping( $order_item ); + } + + if ( -1 !== $quantity ) { + if ( $quantity > $quantity_left ) { + $quantity = $quantity_left; + } + } else { + $quantity = $quantity_left; + } + } + + if ( $quantity <= 0 ) { + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_item_id', _x( 'This order item does not need shipping/returning.', 'shipments', 'woocommerce-germanized' ), 400 ); + } + + if ( $shipment->get_item_by_order_item_id( $item->get_order_item_id() ) ) { + throw new \WC_REST_Exception( 'woocommerce_gzd_rest_invalid_item_id', _x( 'The order item is already associated with another item.', 'shipments', 'woocommerce-germanized' ), 400 ); + } + + $item->sync( array( 'quantity' => $quantity ) ); + } else { + if ( $order_shipment = $shipment->get_order_shipment() ) { + $quantity = isset( $posted['quantity'] ) ? absint( wp_unslash( $posted['quantity'] ) ) : $item->get_quantity(); + $quantity_left = 0; + + if ( 'return' === $shipment->get_type() ) { + $quantity_left = $order_shipment->get_item_quantity_left_for_returning( + $item->get_order_item_id(), + array( + 'exclude_current_shipment' => true, + 'shipment_id' => $shipment->get_id(), + ) + ); + } elseif ( $order_item = $item->get_order_item() ) { + $quantity_left = $order_shipment->get_item_quantity_left_for_shipping( + $order_item, + array( + 'exclude_current_shipment' => true, + 'shipment_id' => $shipment->get_id(), + ) + ); + } + + if ( $quantity > $quantity_left ) { + $quantity = $quantity_left; + } + + if ( $quantity <= 0 ) { + $shipment->remove_item( $item->get_id() ); + return; + } + + $shipment->update_item_quantity( $item->get_id(), $quantity ); + } + } + + $props_to_set = array( + 'name', + 'product_id', + 'sku', + 'total', + 'subtotal', + 'weight', + 'hs_code', + 'manufacture_country', + 'return_reason_code', + ); + + foreach ( $props_to_set as $prop ) { + $setter = "set_{$prop}"; + + if ( isset( $posted[ $prop ] ) && is_callable( array( $item, $setter ) ) ) { + $item->{$setter}( wc_clean( wp_unslash( $posted[ $prop ] ) ) ); + } + } + + if ( isset( $posted['dimensions'] ) && is_array( $posted['dimensions'] ) ) { + if ( isset( $posted['dimensions']['length'] ) ) { + $item->set_length( wc_clean( wp_unslash( $posted['dimensions']['length'] ) ) ); + } + if ( isset( $posted['dimensions']['width'] ) ) { + $item->set_width( wc_clean( wp_unslash( $posted['dimensions']['width'] ) ) ); + } + if ( isset( $posted['dimensions']['height'] ) ) { + $item->set_height( wc_clean( wp_unslash( $posted['dimensions']['height'] ) ) ); + } + } + + if ( isset( $posted['attributes'] ) && is_array( $posted['attributes'] ) ) { + $attributes_to_save = array(); + + foreach ( $posted['attributes'] as $attribute ) { + $attribute = wc_clean( wp_unslash( $attribute ) ); + $attribute = wp_parse_args( + $attribute, + array( + 'key' => '', + 'value' => '', + 'label' => '', + 'order_item_meta_id' => 0, + ) + ); + + $attributes_to_save[] = array_intersect_key( $attribute, array_flip( array( 'key', 'value', 'label', 'order_item_meta_id' ) ) ); + } + + $item->set_attributes( $attributes_to_save ); + } + + if ( ! empty( $posted['meta_data'] ) && is_array( $posted['meta_data'] ) ) { + foreach ( $posted['meta_data'] as $meta ) { + $meta = wc_clean( wp_unslash( $meta ) ); + + if ( isset( $meta['key'] ) ) { + $value = isset( $meta['value'] ) ? $meta['value'] : null; + $item->update_meta_data( $meta['key'], $value, isset( $meta['id'] ) ? $meta['id'] : '' ); + } + } + } + + do_action( 'woocommerce_gzd_rest_set_shipment_item', $item, $posted ); + + // If creating the shipment, add the item to it. + if ( 'create' === $action ) { + $shipment->add_item( $item ); + } else { + $item->save(); + } + } + + /** + * Helper method to check if the resource ID associated with the provided item is null. + * Items can be deleted by setting the resource ID to null. + * + * @param array $item Item provided in the request body. + * @return bool True if the item resource ID is null, false otherwise. + */ + protected function item_is_null( $item ) { + $keys = array( 'order_item_id', 'name', 'product_id' ); + + foreach ( $keys as $key ) { + if ( array_key_exists( $key, $item ) && is_null( $item[ $key ] ) ) { + return true; + } + } + + return false; + } + + /** + * Save an object data. + * + * @since 3.0.0 + * @param WP_REST_Request $request Full details about the request. + * @param bool $creating If is creating a new object. + * @return Shipment|WP_Error + */ + protected function save_object( $request, $creating = false ) { + try { + $object = $this->prepare_object_for_database( $request, $creating ); + + if ( is_wp_error( $object ) ) { + return $object; + } + + $object->save(); + + return $this->get_object( $object->get_id() ); + } catch ( \WC_Data_Exception $e ) { + return new WP_Error( $e->getErrorCode(), $e->getMessage(), $e->getErrorData() ); + } catch ( \WC_REST_Exception $e ) { + return new WP_Error( $e->getErrorCode(), $e->getMessage(), array( 'status' => $e->getCode() ) ); + } + } + + /** + * Create a single item. + * + * @param WP_REST_Request $request Full details about the request. + * @return WP_Error|WP_REST_Response + */ + public function create_item( $request ) { + if ( ! empty( $request['id'] ) ) { + /* translators: %s: post type */ + return new WP_Error( 'woocommerce_gzd_rest_shipment_exists', _x( 'Cannot create existing shipment.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 400 ) ); + } + + $object = $this->save_object( $request, true ); + + if ( is_wp_error( $object ) ) { + return $object; + } + + try { + $this->update_additional_fields_for_object( $object, $request ); + + /** + * Fires after a single object is created or updated via the REST API. + * + * @param Shipment $shipment Inserted object. + * @param WP_REST_Request $request Request object. + * @param boolean $creating True when creating object, false when updating. + */ + do_action( 'woocommerce_gzd_rest_insert_shipment_object', $object, $request, true ); + } catch ( \WC_Data_Exception $e ) { + $object->delete(); + return new WP_Error( $e->getErrorCode(), $e->getMessage(), $e->getErrorData() ); + } catch ( \WC_REST_Exception $e ) { + $object->delete(); + return new WP_Error( $e->getErrorCode(), $e->getMessage(), array( 'status' => $e->getCode() ) ); + } + + $request->set_param( 'context', 'edit' ); + $response = $this->prepare_object_for_response( $object, $request ); + $response = rest_ensure_response( $response ); + $response->set_status( 201 ); + $response->header( 'Location', rest_url( sprintf( '/%s/%s/%d', $this->namespace, $this->rest_base, $object->get_id() ) ) ); + + return $response; + } + + /** + * Update a single post. + * + * @param WP_REST_Request $request Full details about the request. + * @return WP_Error|WP_REST_Response + */ + public function update_item( $request ) { + $shipment = $this->get_object( (int) $request['id'] ); + + if ( ! $shipment || 0 === $shipment->get_id() ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_invalid_id', _x( 'Invalid ID.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 400 ) ); + } + + $shipment = $this->save_object( $request, false ); + + if ( is_wp_error( $shipment ) ) { + return $shipment; + } + + try { + $this->update_additional_fields_for_object( $shipment, $request ); + + /** + * Fires after a single shipment is created or updated via the REST API. + * + * @param Shipment $shipment Inserted object. + * @param WP_REST_Request $request Request object. + * @param boolean $creating True when creating object, false when updating. + */ + do_action( 'woocommerce_gzd_rest_insert_shipment_object', $shipment, $request, false ); + } catch ( \WC_Data_Exception $e ) { + return new WP_Error( $e->getErrorCode(), $e->getMessage(), $e->getErrorData() ); + } catch ( \WC_REST_Exception $e ) { + return new WP_Error( $e->getErrorCode(), $e->getMessage(), array( 'status' => $e->getCode() ) ); + } + + $request->set_param( 'context', 'edit' ); + $response = $this->prepare_object_for_response( $shipment, $request ); + + return rest_ensure_response( $response ); + } + + /** + * Prepares the object for the REST response. + * + * @since 3.0.0 + * @param Shipment $shipment Object data. + * @param WP_REST_Request $request Request object. + * @return WP_Error|WP_REST_Response Response object on success, or WP_Error object on failure. + */ + protected function prepare_object_for_response( $shipment, $request ) { + $context = ! empty( $request['context'] ) ? $request['context'] : 'view'; + $this->request = $request; + $data = self::prepare_shipment( $shipment, $context ); + $data = $this->add_additional_fields_to_object( $data, $request ); + $data = $this->filter_response_by_context( $data, $context ); + $response = rest_ensure_response( $data ); + + $response->add_links( $this->prepare_links( $shipment, $request ) ); + + /** + * Filter the shipment data for a response. + * + * @param WP_REST_Response $response The response object. + * @param Shipment $shipment Object data. + * @param WP_REST_Request $request Request object. + */ + return apply_filters( 'woocommerce_gzd_rest_prepare_shipment_object', $response, $shipment, $request ); + } + + /** + * Prepare links for the request. + * + * @param Shipment $shipment Object data. + * @param WP_REST_Request $request Request object. + * @return array Links for the given post. + */ + protected function prepare_links( $shipment, $request ) { + $links = array( + 'self' => array( + 'href' => rest_url( sprintf( '/%s/%s/%d', $this->namespace, $this->rest_base, $shipment->get_id() ) ), + ), + 'collection' => array( + 'href' => rest_url( sprintf( '/%s/%s', $this->namespace, $this->rest_base ) ), + ), + ); + + return $links; + } + + /** + * Get a single item. + * + * @param WP_REST_Request $request Full details about the request. + * @return WP_Error|WP_REST_Response + */ + public function get_item( $request ) { + $object = $this->get_object( (int) $request['id'] ); + + if ( ! $object || 0 === $object->get_id() ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_invalid_id', _x( 'Invalid ID.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 404 ) ); + } + + $data = $this->prepare_object_for_response( $object, $request ); + $response = rest_ensure_response( $data ); + + return $response; + } + + /** + * Checks if a given request has access to delete a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has access to delete the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function delete_item_permissions_check( $request ) { + if ( ! $this->check_permissions( 'shipment', 'delete', $request['id'] ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_delete', _x( 'Sorry, you are not allowed to delete this resource.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + /** + * Deletes one item from the collection. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure. + * @since 4.7.0 + */ + public function delete_item( $request ) { + $force = (bool) $request['force']; + $shipment = $this->get_shipment( (int) $request['id'] ); + + if ( ! $shipment ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_invalid_id', _x( 'Invalid ID.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 404 ) ); + } + + if ( ! $shipment->delete( $force ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_delete', _x( 'The shipment cannot be deleted.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 500 ) ); + } + + return rest_ensure_response( self::prepare_shipment( $shipment ) ); + } + + /** + * Checks if a given request has access to get a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has read access for the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function get_label_permissions_check( $request ) { + if ( ! $this->check_permissions( 'shipment_label', 'read', $request['id'] ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_view', _x( 'Sorry, you are not allowed to view this resource.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + /** + * Retrieves one item from the collection. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return WP_REST_Response|WP_Error Response object on success, or WP_Error object on failure. + * @since 4.7.0 + */ + public function get_label( $request ) { + $shipment = $this->get_shipment( (int) $request['id'] ); + + if ( ! $shipment ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_invalid_id', _x( 'Invalid ID.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 404 ) ); + } + + $label = $shipment->get_label(); + + if ( ! $label ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_invalid_id', _x( 'Invalid ID.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 404 ) ); + } + + $label_data = self::prepare_label( $label ); + + return rest_ensure_response( $label_data ); + } + + public function delete_label( $request ) { + $force = (bool) $request['force']; + $shipment = $this->get_shipment( (int) $request['id'] ); + + if ( ! $shipment ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_invalid_id', _x( 'Invalid ID.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 404 ) ); + } + + $label = $shipment->get_label(); + + if ( ! $label || ! $label->delete( $force ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_delete', _x( 'The label cannot be deleted.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 500 ) ); + } + + return rest_ensure_response( self::prepare_label( $label ) ); + } + + public function create_label( $request ) { + $shipment = $this->get_shipment( (int) $request['id'] ); + + if ( ! $shipment ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_invalid_id', _x( 'Invalid ID.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 404 ) ); + } + + if ( ! $shipment->supports_label() || ! $shipment->needs_label() ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_label_exists', _x( 'Label already exists, please delete first.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 404 ) ); + } + + $request->set_param( 'context', 'edit' ); + + $args = wc_clean( wp_unslash( $request['args'] ) ); + $args = empty( $args ) ? false : $args; + $result = $shipment->create_label( $args ); + + if ( is_wp_error( $result ) ) { + $result = wc_gzd_get_shipment_error( $result ); + } + + if ( is_wp_error( $result ) && ! $result->is_soft_error() ) { + $message = implode( ' | ', $result->get_error_messages() ); + + return new WP_Error( 'woocommerce_gzd_rest_shipment_label_create', $message, array( 'status' => 500 ) ); + } + + $label = $shipment->get_label(); + + if ( ! $label ) { + return new WP_Error( 'woocommerce_gzd_rest_shipment_label_create', _x( 'There was an error creating the label.', 'shipments', 'woocommerce-germanized' ), array( 'status' => 500 ) ); + } + + $response = self::prepare_label( $label, $request ); + $response = rest_ensure_response( $response ); + + $response->set_status( 201 ); + $response->header( 'Location', rest_url( sprintf( '/%s/%s/%d/label', $this->namespace, $this->rest_base, $label->get_shipment_id() ) ) ); + + return $response; + } + + /** + * Checks if a given request has access to create a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has access to delete the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function create_label_permissions_check( $request ) { + if ( ! $this->check_permissions( 'shipment_label', 'create' ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_delete', _x( 'Sorry, you are not allowed to create resources.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + /** + * Checks if a given request has access to delete a specific item. + * + * @param WP_REST_Request $request Full details about the request. + * + * @return true|WP_Error True if the request has access to delete the item, WP_Error object otherwise. + * @since 4.7.0 + */ + public function delete_label_permissions_check( $request ) { + if ( ! $this->check_permissions( 'shipment_label', 'delete', $request['id'] ) ) { + return new WP_Error( 'woocommerce_gzd_rest_cannot_delete', _x( 'Sorry, you are not allowed to delete this resource.', 'shipments', 'woocommerce-germanized' ), array( 'status' => rest_authorization_required_code() ) ); + } + + return true; + } + + /** + * Retrieves the item's schema, conforming to JSON Schema. + * + * @return array Item schema data. + * @since 4.7.0 + */ + public function get_item_schema() { + return $this->add_additional_fields_schema( self::get_single_item_schema() ); + } + + /** + * @param Shipment $shipment + * @param string $context + * @param bool|int $dp + * + * @return array + */ + public static function prepare_shipment( $shipment, $context = 'view', $dp = false ) { + $item_data = array(); + + foreach ( $shipment->get_items() as $item ) { + $item_data[] = array( + 'id' => $item->get_id(), + 'name' => $item->get_name( $context ), + 'order_item_id' => $item->get_order_item_id( $context ), + 'product_id' => $item->get_product_id( $context ), + 'sku' => $item->get_sku( $context ), + 'quantity' => $item->get_quantity( $context ), + 'total' => wc_format_decimal( $item->get_total( $context ), $dp ), + 'subtotal' => wc_format_decimal( $item->get_subtotal( $context ), $dp ), + 'weight' => wc_format_decimal( $item->get_weight( $context ), $dp ), + 'dimensions' => array( + 'length' => wc_format_decimal( $item->get_length( $context ), $dp ), + 'width' => wc_format_decimal( $item->get_width( $context ), $dp ), + 'height' => wc_format_decimal( $item->get_height( $context ), $dp ), + ), + 'hs_code' => $item->get_hs_code( $context ), + 'manufacture_country' => $item->get_manufacture_country( $context ), + 'attributes' => $item->get_attributes( $context ), + 'meta_data' => $item->get_meta_data(), + ); + } + + return array( + 'id' => $shipment->get_id(), + 'shipment_number' => $shipment->get_shipment_number(), + 'date_created' => wc_rest_prepare_date_response( $shipment->get_date_created( $context ), false ), + 'date_created_gmt' => wc_rest_prepare_date_response( $shipment->get_date_created( $context ) ), + 'date_sent' => wc_rest_prepare_date_response( $shipment->get_date_sent( $context ), false ), + 'date_sent_gmt' => wc_rest_prepare_date_response( $shipment->get_date_sent( $context ) ), + 'est_delivery_date' => wc_rest_prepare_date_response( $shipment->get_est_delivery_date( $context ), false ), + 'est_delivery_date_gmt' => wc_rest_prepare_date_response( $shipment->get_est_delivery_date( $context ) ), + 'total' => wc_format_decimal( $shipment->get_total( $context ), $dp ), + 'subtotal' => wc_format_decimal( $shipment->get_subtotal( $context ), $dp ), + 'additional_total' => wc_format_decimal( $shipment->get_additional_total( $context ), $dp ), + 'order_id' => $shipment->get_order_id( $context ), + 'order_number' => $shipment->get_order_number(), + 'weight' => wc_format_decimal( $shipment->get_weight( $context ), $dp ), + 'content_weight' => wc_format_decimal( $shipment->get_content_weight(), $dp ), + 'weight_unit' => $shipment->get_weight_unit( $context ), + 'packaging_id' => $shipment->get_packaging_id( $context ), + 'packaging_weight' => $shipment->get_packaging_weight( $context ), + 'status' => $shipment->get_status( $context ), + 'tracking_id' => $shipment->get_tracking_id( $context ), + 'tracking_url' => $shipment->get_tracking_url(), + 'shipping_provider' => $shipment->get_shipping_provider( $context ), + 'content_dimensions' => array( + 'length' => wc_format_decimal( $shipment->get_content_length(), $dp ), + 'width' => wc_format_decimal( $shipment->get_content_width(), $dp ), + 'height' => wc_format_decimal( $shipment->get_content_height(), $dp ), + ), + 'dimensions' => array( + 'length' => wc_format_decimal( $shipment->get_length( $context ), $dp ), + 'width' => wc_format_decimal( $shipment->get_width( $context ), $dp ), + 'height' => wc_format_decimal( $shipment->get_height( $context ), $dp ), + ), + 'package_dimensions' => array( + 'length' => wc_format_decimal( $shipment->get_package_length(), $dp ), + 'width' => wc_format_decimal( $shipment->get_package_width(), $dp ), + 'height' => wc_format_decimal( $shipment->get_package_height(), $dp ), + ), + 'dimension_unit' => $shipment->get_dimension_unit( $context ), + 'address' => $shipment->get_address( $context ), + 'sender_address' => 'return' === $shipment->get_type() ? $shipment->get_sender_address( $context ) : array(), + 'is_customer_requested' => 'return' === $shipment->get_type() ? $shipment->get_is_customer_requested( $context ) : false, + 'items' => $item_data, + 'meta_data' => $shipment->get_meta_data(), + ); + } + + /** + * @param Label $label + * + * @return + */ + private static function get_label_file( $label, $file_type = '' ) { + $result = array( + 'file' => '', + 'filename' => $label->get_filename( $file_type ), + 'path' => $label->get_path( 'view', $file_type ), + 'type' => $file_type, + ); + + if ( $file = $label->get_file( $file_type ) ) { + try { + $content = file_get_contents( $file ); // phpcs:ignore WordPress.WP.AlternativeFunctions.file_get_contents_file_get_contents + $result['file'] = chunk_split( base64_encode( $content ) ); // phpcs:ignore WordPress.PHP.DiscouragedPHPFunctions.obfuscation_base64_encode + } catch ( \Exception $ex ) { + $result['file'] = ''; + } + } + + return $result; + } + + /** + * @param Label $label + * @param string $context + * @param bool|int $dp + * + * @return array + */ + public static function prepare_label( $label, $context = 'view', $dp = false ) { + $label_data = array( + 'id' => $label->get_id(), + 'date_created' => wc_rest_prepare_date_response( $label->get_date_created( $context ), false ), + 'date_created_gmt' => wc_rest_prepare_date_response( $label->get_date_created( $context ) ), + 'weight' => wc_format_decimal( $label->get_weight( $context ), $dp ), + 'net_weight' => wc_format_decimal( $label->get_net_weight( $context ), $dp ), + 'dimensions' => array( + 'length' => wc_format_decimal( $label->get_length( $context ), $dp ), + 'width' => wc_format_decimal( $label->get_width( $context ), $dp ), + 'height' => wc_format_decimal( $label->get_height( $context ), $dp ), + ), + 'shipment_id' => $label->get_shipment_id( $context ), + 'parent_id' => $label->get_parent_id( $context ), + 'product_id' => $label->get_product_id( $context ), + 'number' => $label->get_number( $context ), + 'type' => $label->get_type(), + 'shipping_provider' => $label->get_shipping_provider( $context ), + 'created_via' => $label->get_created_via( $context ), + 'services' => $label->get_services( $context ), + 'additional_file_types' => array(), + 'files' => array( self::get_label_file( $label ) ), + ); + + foreach ( $label->get_additional_file_types() as $file_type ) { + if ( 'default' === $file_type ) { + continue; + } + + $label_file = self::get_label_file( $label, $file_type ); + + if ( ! empty( $label_file['file'] ) ) { + $label_data['files'][] = $label_file; + $label_data['additional_file_types'][] = $file_type; + } + } + + return $label_data; + } + + /** + * Get the query params for collections. + * + * @return array + */ + public function get_collection_params() { + $params = parent::get_collection_params(); + + $params['context']['default'] = 'view'; + + $params['offset'] = array( + 'description' => _x( 'Offset the result set by a specific number of items.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'sanitize_callback' => 'absint', + 'validate_callback' => 'rest_validate_request_arg', + ); + $params['order'] = array( + 'default' => 'desc', + 'description' => _x( 'Order sort attribute ascending or descending.', 'shipments', 'woocommerce-germanized' ), + 'enum' => array( 'asc', 'desc' ), + 'sanitize_callback' => 'sanitize_key', + 'type' => 'string', + 'validate_callback' => 'rest_validate_request_arg', + ); + $params['orderby'] = array( + 'default' => 'date_created', + 'description' => _x( 'Sort collection by object attribute.', 'shipments', 'woocommerce-germanized' ), + 'enum' => array( + 'country', + 'status', + 'tracking_id', + 'date_created', + 'order_id', + 'weight', + ), + 'sanitize_callback' => 'sanitize_key', + 'type' => 'string', + 'validate_callback' => 'rest_validate_request_arg', + ); + $params['order_id'] = array( + 'description' => _x( 'Limit result set to shipments belonging to a certain order id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'sanitize_callback' => 'absint', + 'validate_callback' => 'rest_validate_request_arg', + ); + $params['status'] = array( + 'description' => _x( 'Limit result set to shipments having a certain status.', 'shipments', 'woocommerce-germanized' ), + 'enum' => self::get_shipment_statuses(), + 'sanitize_callback' => 'sanitize_key', + 'type' => 'string', + 'validate_callback' => 'rest_validate_request_arg', + ); + $params['type'] = array( + 'description' => _x( 'Limit result set to shipments of a certain type.', 'shipments', 'woocommerce-germanized' ), + 'default' => 'simple', + 'enum' => wc_gzd_get_shipment_types(), + 'sanitize_callback' => 'sanitize_key', + 'type' => 'string', + 'validate_callback' => 'rest_validate_request_arg', + ); + return $params; + } + + /** + * Get the schema of a single shipment + * + * @return array + */ + public static function get_single_item_schema() { + $weight_unit = get_option( 'woocommerce_weight_unit' ); + $dimension_unit = get_option( 'woocommerce_dimension_unit' ); + + return array( + 'description' => _x( 'Single shipment.', 'shipment', 'woocommerce-germanized' ), + 'context' => array( 'view', 'edit' ), + 'readonly' => false, + 'type' => 'object', + 'properties' => array( + 'id' => array( + 'description' => _x( 'Shipment ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'shipment_number' => array( + 'description' => _x( 'Shipment number.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'order_id' => array( + 'description' => _x( 'Shipment order id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'order_number' => array( + 'description' => _x( 'Shipment order number.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'status' => array( + 'description' => _x( 'Shipment status.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'enum' => self::get_shipment_statuses(), + ), + 'tracking_id' => array( + 'description' => _x( 'Shipment tracking id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'tracking_url' => array( + 'description' => _x( 'Shipment tracking url.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'shipping_provider' => array( + 'description' => _x( 'Shipment shipping provider.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'date_created' => array( + 'description' => _x( "The date the shipment was created, in the site's timezone.", 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'date_created_gmt' => array( + 'description' => _x( 'The date the shipment was created, as GMT.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'date_sent' => array( + 'description' => _x( "The date the shipment was sent, in the site's timezone.", 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'date_sent_gmt' => array( + 'description' => _x( 'The date the shipment was sent, as GMT.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'est_delivery_date' => array( + 'description' => _x( "The estimated delivery date of the shipment, in the site's timezone.", 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + ), + 'est_delivery_date_gmt' => array( + 'description' => _x( 'The estimated delivery date of the shipment, as GMT.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + ), + 'type' => array( + 'description' => _x( 'Shipment type, e.g. simple or return.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'enum' => wc_gzd_get_shipment_types(), + ), + 'is_customer_requested' => array( + 'description' => _x( 'Return shipment is requested by customer.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'boolean', + 'context' => array( 'view', 'edit' ), + ), + 'sender_address' => array( + 'description' => _x( 'Return sender address.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'properties' => array( + 'first_name' => array( + 'description' => _x( 'First name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'last_name' => array( + 'description' => _x( 'Last name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'company' => array( + 'description' => _x( 'Company name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'address_1' => array( + 'description' => _x( 'Address line 1', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'address_2' => array( + 'description' => _x( 'Address line 2', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'city' => array( + 'description' => _x( 'City name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'state' => array( + 'description' => _x( 'ISO code or name of the state, province or district.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'postcode' => array( + 'description' => _x( 'Postal code.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'country' => array( + 'description' => _x( 'Country code in ISO 3166-1 alpha-2 format.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'customs_reference_number' => array( + 'description' => _x( 'Customs reference number.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + 'weight' => array( + 'description' => _x( 'Shipment weight.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'content_weight' => array( + 'description' => _x( 'Shipment content weight.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'content_dimensions' => array( + 'description' => _x( 'Shipment content dimensions.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + 'properties' => array( + 'length' => array( + 'description' => _x( 'Shipment content length.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'width' => array( + 'description' => _x( 'Shipment content width.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'height' => array( + 'description' => _x( 'Shipment content height.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + ), + ), + 'weight_unit' => array( + 'description' => _x( 'Shipment weight unit.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'default' => $weight_unit, + ), + 'packaging_id' => array( + 'description' => _x( 'Shipment packaging id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + ), + 'packaging_weight' => array( + 'description' => _x( 'Shipment packaging weight.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'total' => array( + 'description' => _x( 'Shipment total.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'subtotal' => array( + 'description' => _x( 'Shipment subtotal.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'additional_total' => array( + 'description' => _x( 'Shipment additional total.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'version' => array( + 'description' => _x( 'Shipment version.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'shipping_method' => array( + 'description' => _x( 'Shipment shipping method.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'dimensions' => array( + 'description' => _x( 'Shipment dimensions.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'properties' => array( + 'length' => array( + 'description' => _x( 'Shipment length.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'width' => array( + 'description' => _x( 'Shipment width.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'height' => array( + 'description' => _x( 'Shipment height.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + 'package_dimensions' => array( + 'description' => _x( 'Shipment package dimensions.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + 'properties' => array( + 'length' => array( + 'description' => _x( 'Shipment package length.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'readonly' => true, + 'context' => array( 'view', 'edit' ), + ), + 'width' => array( + 'description' => _x( 'Shipment package width.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'readonly' => true, + 'context' => array( 'view', 'edit' ), + ), + 'height' => array( + 'description' => _x( 'Shipment package height.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'readonly' => true, + 'context' => array( 'view', 'edit' ), + ), + ), + ), + 'dimension_unit' => array( + 'description' => _x( 'Shipment dimension unit.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'default' => $dimension_unit, + ), + 'address' => array( + 'description' => _x( 'Shipping address.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'properties' => array( + 'first_name' => array( + 'description' => _x( 'First name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'last_name' => array( + 'description' => _x( 'Last name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'company' => array( + 'description' => _x( 'Company name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'address_1' => array( + 'description' => _x( 'Address line 1', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'address_2' => array( + 'description' => _x( 'Address line 2', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'city' => array( + 'description' => _x( 'City name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'state' => array( + 'description' => _x( 'ISO code or name of the state, province or district.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'postcode' => array( + 'description' => _x( 'Postal code.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'country' => array( + 'description' => _x( 'Country code in ISO 3166-1 alpha-2 format.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'customs_reference_number' => array( + 'description' => _x( 'Customs reference number.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + 'meta_data' => array( + 'description' => _x( 'Meta data.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'items' => array( + 'type' => 'object', + 'properties' => array( + 'id' => array( + 'description' => _x( 'Meta ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'key' => array( + 'description' => _x( 'Meta key.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'value' => array( + 'description' => _x( 'Meta value.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'mixed', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + ), + 'items' => array( + 'description' => _x( 'Shipment items.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'items' => array( + 'type' => 'object', + 'properties' => array( + 'id' => array( + 'description' => _x( 'Item ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'name' => array( + 'description' => _x( 'Item name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'mixed', + 'context' => array( 'view', 'edit' ), + ), + 'order_item_id' => array( + 'description' => _x( 'Order Item ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + ), + 'product_id' => array( + 'description' => _x( 'Product ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'mixed', + 'context' => array( 'view', 'edit' ), + ), + 'quantity' => array( + 'description' => _x( 'Quantity.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + ), + 'weight' => array( + 'description' => _x( 'Item weight.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'sku' => array( + 'description' => _x( 'Item SKU.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'total' => array( + 'description' => _x( 'Item total.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'subtotal' => array( + 'description' => _x( 'Item subtotal.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'hs_code' => array( + 'description' => _x( 'Item HS Code (customs).', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'manufacture_country' => array( + 'description' => _x( 'Item country of manufacture in ISO 3166-1 alpha-2 format.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'dimensions' => array( + 'description' => _x( 'Item dimensions.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'properties' => array( + 'length' => array( + 'description' => _x( 'Item length.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'width' => array( + 'description' => _x( 'Item width.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'height' => array( + 'description' => _x( 'Item height.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + 'attributes' => array( + 'description' => _x( 'Item attributes.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'properties' => array( + 'key' => array( + 'description' => _x( 'Attribute key.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'value' => array( + 'description' => _x( 'Attribute value.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'label' => array( + 'description' => _x( 'Attribute label.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'order_item_meta_id' => array( + 'description' => _x( 'Order item meta id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + 'meta_data' => array( + 'description' => _x( 'Shipment item meta data.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'items' => array( + 'type' => 'object', + 'properties' => array( + 'id' => array( + 'description' => _x( 'Meta ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'key' => array( + 'description' => _x( 'Meta key.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'value' => array( + 'description' => _x( 'Meta value.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'mixed', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + ), + ), + ), + ), + ), + ); + } + + public function get_public_item_label_schema() { + return array( + 'description' => _x( 'Shipment label.', 'shipment', 'woocommerce-germanized' ), + 'context' => array( 'view', 'edit' ), + 'readonly' => false, + 'type' => 'object', + 'properties' => array( + 'id' => array( + 'description' => _x( 'Label ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'date_created' => array( + 'description' => _x( "The date the label was created, in the site's timezone.", 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'date_created_gmt' => array( + 'description' => _x( 'The date the label was created, as GMT.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'date-time', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'shipment_id' => array( + 'description' => _x( 'Shipment id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'parent_id' => array( + 'description' => _x( 'Parent id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + ), + 'product_id' => array( + 'description' => _x( 'Label product id.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'number' => array( + 'description' => _x( 'Label number.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'shipping_provider' => array( + 'description' => _x( 'Shipping provider.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'weight' => array( + 'description' => _x( 'Weight.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'net_weight' => array( + 'description' => _x( 'Net weight.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'created_via' => array( + 'description' => _x( 'Created via.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'is_trackable' => array( + 'description' => _x( 'Is trackable?', 'shipments', 'woocommerce-germanized' ), + 'type' => 'boolean', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'additional_file_types' => array( + 'description' => _x( 'Additional file types', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + 'items' => array( + 'type' => 'string', + ), + ), + 'files' => array( + 'description' => _x( 'Label file data.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + 'items' => array( + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + 'type' => 'object', + 'properties' => array( + 'path' => array( + 'description' => _x( 'File path.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'filename' => array( + 'description' => _x( 'File name.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'file' => array( + 'description' => _x( 'The file data (base64 encoded).', 'shipments', 'woocommerce-germanized' ), + 'type' => 'binary', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'type' => array( + 'description' => _x( 'File type.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + ), + ), + ), + 'type' => array( + 'description' => _x( 'Label type, e.g. simple or return.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'dimensions' => array( + 'description' => _x( 'Label dimensions.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'object', + 'context' => array( 'view', 'edit' ), + 'properties' => array( + 'length' => array( + 'description' => _x( 'Label length.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'width' => array( + 'description' => _x( 'Label width.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'height' => array( + 'description' => _x( 'Label height.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + 'services' => array( + 'description' => _x( 'Label services.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'items' => array( + 'type' => 'string', + ), + ), + 'meta_data' => array( + 'description' => _x( 'Label meta data.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'array', + 'context' => array( 'view', 'edit' ), + 'items' => array( + 'type' => 'object', + 'properties' => array( + 'id' => array( + 'description' => _x( 'Meta ID.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'integer', + 'context' => array( 'view', 'edit' ), + 'readonly' => true, + ), + 'key' => array( + 'description' => _x( 'Meta key.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'string', + 'context' => array( 'view', 'edit' ), + ), + 'value' => array( + 'description' => _x( 'Meta value.', 'shipments', 'woocommerce-germanized' ), + 'type' => 'mixed', + 'context' => array( 'view', 'edit' ), + ), + ), + ), + ), + ), + ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ReturnReason.php b/packages/woocommerce-germanized-shipments/src/ReturnReason.php new file mode 100644 index 000000000..2b89c904c --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ReturnReason.php @@ -0,0 +1,46 @@ + '', + 'reason' => '', + 'order' => 0, + ) + ); + + $this->args = $args; + } + + public function get_reason() { + return $this->args['reason']; + } + + public function get_code() { + return $this->args['code']; + } + + public function get_name() { + return $this->get_code(); + } + + public function get_order() { + return absint( $this->args['order'] ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ReturnShipment.php b/packages/woocommerce-germanized-shipments/src/ReturnShipment.php new file mode 100644 index 000000000..64ad411fe --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ReturnShipment.php @@ -0,0 +1,478 @@ + 0, + 'is_customer_requested' => false, + 'sender_address' => array(), + ); + + /** + * Returns the shipment type. + * + * @return string + */ + public function get_type() { + return 'return'; + } + + /** + * Returns the order id belonging to the shipment. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return integer + */ + public function get_order_id( $context = 'view' ) { + return $this->get_prop( 'order_id', $context ); + } + + /** + * Returns whether the current return was requested by a customer or not. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return boolean + */ + public function get_is_customer_requested( $context = 'view' ) { + return $this->get_prop( 'is_customer_requested', $context ); + } + + public function is_customer_requested() { + return $this->get_is_customer_requested(); + } + + public function confirm_customer_request() { + if ( $this->is_customer_requested() && $this->has_status( 'requested' ) ) { + + $this->set_status( 'processing' ); + + if ( $this->save() ) { + /** + * Action that fires after a return request has been confirmed to the customer. + * + * @param integer $shipment_id The return shipment id. + * @param ReturnShipment $shipment The return shipment object. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_return_shipment_customer_confirmed', $this->get_id(), $this ); + + return true; + } else { + return false; + } + } + + return false; + } + + /** + * Returns the address of the sender e.g. customer. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string[] + */ + public function get_sender_address( $context = 'view' ) { + return $this->get_prop( 'sender_address', $context ); + } + + /** + * Set shipment order id. + * + * @param string $order_id The order id. + */ + public function set_order_id( $order_id ) { + // Reset order object + $this->order = null; + + $this->set_prop( 'order_id', absint( $order_id ) ); + } + + /** + * Set if the current return was requested by the customer or not. + * + * @param string $is_requested Whether or not it is requested by the customer. + */ + public function set_is_customer_requested( $is_requested ) { + $this->set_prop( 'is_customer_requested', wc_string_to_bool( $is_requested ) ); + } + + /** + * Set shipment order. + * + * @param Order $order_shipment The order shipment. + */ + public function set_order_shipment( &$order_shipment ) { + $this->order_shipment = $order_shipment; + } + + /** + * Returns shipment order. + * + * @return Order|null The order shipment. + */ + public function get_order_shipment() { + if ( is_null( $this->order_shipment ) ) { + $order = $this->get_order(); + $this->order_shipment = ( $order ? wc_gzd_get_shipment_order( $order ) : false ); + } + + return $this->order_shipment; + } + + /** + * Returns the shippable item count. + * + * @return int + */ + public function get_shippable_item_count() { + + if ( $order_shipment = $this->get_order_shipment() ) { + return $order_shipment->get_returnable_item_count(); + } + + return 0; + } + + /** + * Tries to fetch the order for the current shipment. + * + * @return bool|WC_Order|null + */ + public function get_order() { + if ( is_null( $this->order ) ) { + $this->order = ( $this->get_order_id() > 0 ? wc_get_order( $this->get_order_id() ) : false ); + } + + return $this->order; + } + + /** + * Returns available shipment methods by checking the corresponding order. + * + * @return string[] + */ + public function get_available_shipping_methods() { + $methods = array(); + $methods[ $this->get_shipping_method() ] = ''; + + return $methods; + } + + /** + * Returns a sender address prop. + * + * @param string $prop + * @param string $context + * + * @return null|string + */ + protected function get_sender_address_prop( $prop, $context = 'view' ) { + $value = null; + + if ( isset( $this->changes['sender_address'][ $prop ] ) || isset( $this->data['sender_address'][ $prop ] ) ) { + $value = isset( $this->changes['sender_address'][ $prop ] ) ? $this->changes['sender_address'][ $prop ] : $this->data['sender_address'][ $prop ]; + + if ( 'view' === $context ) { + /** + * Filter to adjust a Shipment's return sender address property e.g. first_name. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. `$prop` refers to the actual address property e.g. first_name. + * + * Example hook name: woocommerce_gzd_return_shipment_get_sender_address_first_name + * + * @param string $value The address property value. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $value = apply_filters( "{$this->get_hook_prefix()}sender_address_{$prop}", $value, $this ); + } + } + + return $value; + } + + /** + * Set sender address. + * + * @param string[] $address The address props. + */ + public function set_sender_address( $address ) { + $this->set_prop( 'sender_address', empty( $address ) ? array() : (array) $address ); + } + + /** + * Syncs the return shipment with the corresponding parent shipment. + * + * @param array $args + * + * @return bool + */ + public function sync( $args = array() ) { + try { + if ( ! $order_shipment = $this->get_order_shipment() ) { + throw new Exception( _x( 'Invalid shipment order', 'shipments', 'woocommerce-germanized' ) ); + } + + $return_address = wc_gzd_get_shipment_return_address( $order_shipment ); + $order = $order_shipment->get_order(); + + /** + * Make sure that manually adjusted providers are not overridden by syncing. + */ + $default_provider = $order_shipment->get_default_return_shipping_provider(); + $provider = $this->get_shipping_provider( 'edit' ); + $sender_address_data = array_merge( + ( $order->has_shipping_address() ? $order->get_address( 'shipping' ) : $order->get_address( 'billing' ) ), + array( + 'email' => $order->get_billing_email(), + 'phone' => $order->get_billing_phone(), + ) + ); + + // Prefer shipping phone in case exists + if ( is_callable( array( $order, 'get_shipping_phone' ) ) && $order->get_shipping_phone() ) { + $sender_address_data['phone'] = $order->get_shipping_phone(); + } + + $args = wp_parse_args( + $args, + array( + 'order_id' => $order->get_id(), + 'country' => $return_address['country'], + 'shipping_method' => wc_gzd_get_shipment_order_shipping_method_id( $order ), + 'shipping_provider' => ( ! empty( $provider ) ) ? $provider : $default_provider, + 'address' => $return_address, + 'sender_address' => $sender_address_data, + 'weight' => $this->get_weight( 'edit' ), + 'length' => $this->get_length( 'edit' ), + 'width' => $this->get_width( 'edit' ), + 'height' => $this->get_height( 'edit' ), + ) + ); + + /** + * Filter to allow adjusting the return shipment props synced from the corresponding order. + * + * @param mixed $args The properties in key => value pairs. + * @param ReturnShipment $shipment The shipment object. + * @param Order $order_shipment The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $args = apply_filters( 'woocommerce_gzd_return_shipment_sync_props', $args, $this, $order_shipment ); + + $this->set_props( $args ); + + /** + * Action that fires after a return shipment has been synced. Syncing is used to + * keep the shipment in sync with the corresponding parent shipment. + * + * @param ReturnShipment $shipment The return shipment object. + * @param Order $order_shipment The shipment order object. + * @param array $args Array containing properties in key => value pairs to be updated. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_return_shipment_synced', $this, $order_shipment, $args ); + + } catch ( Exception $e ) { + return false; + } + + return true; + } + + /** + * Keeps items in sync with the parent shipment items. + * Limits quantities and removes non-existent items. + * + * @param array $args + * + * @return bool + */ + public function sync_items( $args = array() ) { + try { + + if ( ! $order_shipment = $this->get_order_shipment() ) { + throw new Exception( _x( 'Invalid shipment order', 'shipments', 'woocommerce-germanized' ) ); + } + + $order = $order_shipment->get_order(); + + $args = wp_parse_args( + $args, + array( + 'items' => array(), + ) + ); + + $available_items = $order_shipment->get_available_items_for_return( + array( + 'shipment_id' => $this->get_id(), + 'exclude_current_shipment' => true, + ) + ); + + foreach ( $available_items as $order_item_id => $item_data ) { + + if ( $item = $order_shipment->get_simple_shipment_item( $order_item_id ) ) { + $quantity = $item_data['max_quantity']; + $return_reason_code = ''; + + if ( ! empty( $args['items'] ) ) { + if ( isset( $args['items'][ $order_item_id ] ) ) { + + if ( is_array( $args['items'][ $order_item_id ] ) ) { + $default_item_data = wp_parse_args( + $args['items'][ $order_item_id ], + array( + 'quantity' => 1, + 'return_reason_code' => '', + ) + ); + } else { + $default_item_data = array( + 'quantity' => absint( $args['items'][ $order_item_id ] ), + 'return_reason_code' => '', + ); + } + + $new_quantity = $default_item_data['quantity']; + $return_reason_code = $default_item_data['return_reason_code']; + + if ( $new_quantity < $quantity ) { + $quantity = $new_quantity; + } + } else { + continue; + } + } + + if ( $quantity <= 0 ) { + continue; + } + + $sync_data = array( + 'quantity' => $quantity, + ); + + if ( ! empty( $return_reason_code ) ) { + $sync_data['return_reason_code'] = $return_reason_code; + } + + if ( ! $shipment_item = $this->get_item_by_order_item_id( $order_item_id ) ) { + $shipment_item = wc_gzd_create_return_shipment_item( $this, $item, $sync_data ); + + $this->add_item( $shipment_item ); + } else { + $shipment_item->sync( $sync_data ); + } + } + } + + foreach ( $this->get_items() as $item ) { + + // Remove non-existent items + if ( ! $shipment_item = $order_shipment->get_simple_shipment_item( $item->get_order_item_id() ) ) { + $this->remove_item( $item->get_id() ); + } + } + + // Sync packaging + $this->sync_packaging(); + + /** + * Action that fires after items of a shipment have been synced. + * + * @param SimpleShipment $shipment The shipment object. + * @param Order $order_shipment The shipment order object. + * @param array $args Array containing additional data e.g. items. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_return_shipment_items_synced', $this, $order_shipment, $args ); + + } catch ( Exception $e ) { + return false; + } + + return true; + } + + /** + * Returns whether the Shipment needs additional items or not. + * + * @param bool|integer[] $available_items + * + * @return bool + */ + public function needs_items( $available_items = false ) { + + if ( ! $available_items && ( $order_shipment = $this->get_order_shipment() ) ) { + $available_items = array_keys( $order_shipment->get_available_items_for_return() ); + } + + return ( $this->is_editable() && ! $this->contains_order_item( $available_items ) ); + } + + /** + * Returns the edit shipment URL. + * + * @return mixed|string|void + */ + public function get_edit_shipment_url() { + /** + * Filter to adjust the edit Shipment admin URL. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_edit_url + * + * @param string $url The URL. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}edit_url", get_admin_url( null, 'post.php?post=' . $this->get_order_id() . '&action=edit&shipment_id=' . $this->get_id() ), $this ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Shipment.php b/packages/woocommerce-germanized-shipments/src/Shipment.php new file mode 100644 index 000000000..1e8aca751 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Shipment.php @@ -0,0 +1,2820 @@ + null, + 'date_sent' => null, + 'status' => '', + 'weight' => '', + 'width' => '', + 'height' => '', + 'length' => '', + 'packaging_weight' => '', + 'weight_unit' => '', + 'dimension_unit' => '', + 'country' => '', + 'address' => array(), + 'tracking_id' => '', + 'shipping_provider' => '', + 'shipping_method' => '', + 'total' => 0, + 'subtotal' => 0, + 'additional_total' => 0, + 'est_delivery_date' => null, + 'packaging_id' => 0, + 'version' => '', + ); + + /** + * Get the shipment if ID is passed, otherwise the shipment is new and empty. + * This class should NOT be instantiated, but the `wc_gzd_get_shipment` function should be used. + * + * @param int|object|Shipment $shipment Shipment to read. + */ + public function __construct( $data = 0 ) { + parent::__construct( $data ); + + if ( $data instanceof Shipment ) { + $this->set_id( absint( $data->get_id() ) ); + } elseif ( is_numeric( $data ) ) { + $this->set_id( $data ); + } + + $this->data_store = WC_Data_Store::load( $this->data_store_name ); + + // If we have an ID, load the user from the DB. + if ( $this->get_id() ) { + try { + $this->data_store->read( $this ); + } catch ( Exception $e ) { + $this->set_id( 0 ); + $this->set_object_read( true ); + } + } else { + $this->set_object_read( true ); + } + } + + public function get_type() { + return ''; + } + + /** + * Merge changes with data and clear. + * Overrides WC_Data::apply_changes. + * + * @since 3.2.0 + */ + public function apply_changes() { + if ( function_exists( 'array_replace' ) ) { + $this->data = array_replace( $this->data, $this->changes ); // phpcs:ignore PHPCompatibility.FunctionUse.NewFunctions.array_replaceFound + } else { // PHP 5.2 compatibility. + foreach ( $this->changes as $key => $change ) { + $this->data[ $key ] = $change; + } + } + $this->changes = array(); + } + + /** + * @return bool|Order + */ + public function get_order_shipment() { + return false; + } + + public function set_order_shipment( &$order_shipment ) {} + + /** + * Return item count (quantities summed up). + * + * @return int + */ + public function get_item_count() { + $items = $this->get_items(); + $quantity = 0; + + foreach ( $items as $item ) { + $quantity += $item->get_quantity(); + } + + return $quantity; + } + + /** + * Prefix for action and filter hooks on data. + * + * @return string + */ + protected function get_hook_prefix() { + return $this->get_general_hook_prefix() . 'get_'; + } + + /** + * Prefix for action and filter hooks on data. + * + * @return string + */ + protected function get_general_hook_prefix() { + $shipment_prefix = 'simple' === $this->get_type() ? '' : $this->get_type() . '_'; + + return "woocommerce_gzd_{$shipment_prefix}shipment_"; + } + + public function is_shipping_domestic() { + return Package::is_shipping_domestic( + $this->get_country(), + array( + 'sender_country' => $this->get_sender_country(), + 'sender_postcode' => $this->get_sender_postcode(), + 'postcode' => $this->get_postcode(), + ) + ); + } + + /** + * Returns true in case the shipment is being shipped inner EU, e.g. + * from a base country inside of the EU to another country inside the EU. + * + * @return bool + */ + public function is_shipping_inner_eu() { + if ( Package::is_shipping_inner_eu_country( + $this->get_country(), + array( + 'sender_country' => $this->get_sender_country(), + 'sender_postcode' => $this->get_sender_postcode(), + 'postcode' => $this->get_postcode(), + ) + ) ) { + return true; + } + + return false; + } + + public function is_shipping_international() { + if ( $this->is_shipping_domestic() || $this->is_shipping_inner_eu() ) { + return false; + } + + return true; + } + + /** + * Return the shipment statuses without gzd- internal prefix. + * + * @param string $context View or edit context. + * @return string + */ + public function get_status( $context = 'view' ) { + $status = $this->get_prop( 'status', $context ); + + if ( empty( $status ) && 'view' === $context ) { + + /** + * Filters the default Shipment status used as fallback. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_default_shipment_status + * + * @param string $status Default fallback status. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $status = apply_filters( "{$this->get_hook_prefix()}}default_shipment_status", 'draft' ); // phpcs:ignore WordPress.NamingConventions.ValidHookName.UseUnderscores + } + + return $status; + } + + /** + * Checks whether the shipment has a specific status or not. + * + * @param string|string[] $status The status to be checked against. + * @return boolean + */ + public function has_status( $status ) { + /** + * Filter to decide whether a Shipment has a certain status or not. + * + * @param boolean $has_status Whether the Shipment has a status or not. + * @param Shipment $this The shipment object. + * @param string $status The status to be checked against. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_has_status', ( is_array( $status ) && in_array( $this->get_status(), $status, true ) ) || $this->get_status() === $status, $this, $status ); + } + + /** + * Return the date this shipment was created. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return WC_DateTime|null object if the date is set or null if there is no date. + */ + public function get_date_created( $context = 'view' ) { + return $this->get_prop( 'date_created', $context ); + } + + /** + * Return the date this shipment was sent. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return WC_DateTime|null object if the date is set or null if there is no date. + */ + public function get_date_sent( $context = 'view' ) { + return $this->get_prop( 'date_sent', $context ); + } + + /** + * Returns the shipment method. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_shipping_method( $context = 'view' ) { + return $this->get_prop( 'shipping_method', $context ); + } + + public function get_shipping_method_instance() { + $method_id = $this->get_shipping_method(); + + if ( is_null( $this->shipping_method_instance ) && ! empty( $method_id ) ) { + $this->shipping_method_instance = wc_gzd_get_shipping_provider_method( $this->get_shipping_method() ); + } + + return is_null( $this->shipping_method_instance ) ? false : $this->shipping_method_instance; + } + + /** + * Returns the shipment weight. In case view context was chosen and weight is not yet set, returns the content weight. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_weight( $context = 'view' ) { + $weight = $this->get_prop( 'weight', $context ); + + if ( 'view' === $context && '' === $weight ) { + return $this->get_content_weight(); + } + + return $weight; + } + + public function get_total_weight() { + $weight = $this->get_weight() + $this->get_packaging_weight(); + + return $weight; + } + + public function get_packaging_weight( $context = 'view' ) { + $weight = $this->get_prop( 'packaging_weight', $context ); + + if ( 'view' === $context && '' === $weight ) { + $weight = wc_format_decimal( 0 ); + + if ( $packaging = $this->get_packaging() ) { + if ( ! empty( $packaging->get_weight() ) ) { + $weight = wc_get_weight( $packaging->get_weight(), $this->get_weight_unit(), wc_gzd_get_packaging_weight_unit() ); + } + } + } + + return $weight; + } + + public function get_items_to_pack() { + if ( ! Package::is_packing_supported() ) { + return $this->get_items(); + } else { + if ( is_null( $this->items_to_pack ) ) { + $this->items_to_pack = array(); + + foreach ( $this->get_items() as $item ) { + for ( $i = 0; $i < $item->get_quantity(); $i++ ) { // phpcs:ignore Generic.CodeAnalysis.ForLoopWithTestFunctionCall.NotAllowed + $box_item = new Packing\ShipmentItem( $item ); + $this->items_to_pack[] = $box_item; + } + } + } + + return apply_filters( "{$this->get_hook_prefix()}items_to_pack", $this->items_to_pack, $this ); + } + } + + /** + * Returns the shipment weight unit. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_weight_unit( $context = 'view' ) { + $unit = $this->get_prop( 'weight_unit', $context ); + + if ( 'view' === $context && '' === $unit ) { + return get_option( 'woocommerce_weight_unit' ); + } + + return $unit; + } + + /** + * Returns the shipment dimension unit. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_dimension_unit( $context = 'view' ) { + $unit = $this->get_prop( 'dimension_unit', $context ); + + if ( 'view' === $context && '' === $unit ) { + return get_option( 'woocommerce_dimension_unit' ); + } + + return $unit; + } + + /** + * Returns the shipment length. In case view context was chosen and length is not yet set, returns the content length. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_length( $context = 'view' ) { + $length = $this->get_prop( 'length', $context ); + + if ( 'view' === $context && '' === $length ) { + return $this->get_content_length(); + } + + return $length; + } + + public function get_package_length() { + $length = $this->get_length(); + + // Older versions did not sync dimensions with packaging dimensions + if ( '' === $this->get_version() ) { + if ( $packaging = $this->get_packaging() ) { + $length = wc_get_dimension( $packaging->get_length(), $this->get_dimension_unit(), wc_gzd_get_packaging_dimension_unit() ); + } + } + + return $length; + } + + public function has_packaging() { + return ( $this->get_packaging_id() > 0 && $this->get_packaging() ) ? true : false; + } + + /** + * Returns the shipment width. In case view context was chosen and width is not yet set, returns the content width. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_width( $context = 'view' ) { + $width = $this->get_prop( 'width', $context ); + + if ( 'view' === $context && '' === $width ) { + return $this->get_content_width(); + } + + return $width; + } + + public function get_package_width() { + $width = $this->get_width(); + + if ( '' === $this->get_version() ) { + if ( $packaging = $this->get_packaging() ) { + $width = wc_get_dimension( $packaging->get_width(), $this->get_dimension_unit(), wc_gzd_get_packaging_dimension_unit() ); + } + } + + return $width; + } + + /** + * Returns the shipment height. In case view context was chosen and height is not yet set, returns the content height. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_height( $context = 'view' ) { + $height = $this->get_prop( 'height', $context ); + + if ( 'view' === $context && '' === $height ) { + return $this->get_content_height(); + } + + return $height; + } + + public function get_package_height() { + $height = $this->get_height(); + + if ( '' === $this->get_version() ) { + if ( $packaging = $this->get_packaging() ) { + $height = wc_get_dimension( $packaging->get_height(), $this->get_dimension_unit(), wc_gzd_get_packaging_dimension_unit() ); + } + } + + return $height; + } + + public function has_dimensions() { + $width = $this->get_width(); + $length = $this->get_length(); + $height = $this->get_height(); + + return ( ! empty( $width ) && ! empty( $length ) && ! empty( $height ) ); + } + + /** + * Returns the calculated weights for included items. + * + * @return float[] + */ + public function get_item_weights() { + if ( is_null( $this->weights ) ) { + $this->weights = array(); + + foreach ( $this->get_items() as $item ) { + $this->weights[ $item->get_id() ] = ( ( $item->get_weight() === '' ? 0 : $item->get_weight() ) * $item->get_quantity() ); + } + + if ( empty( $this->weights ) ) { + $this->weights = array( 0 ); + } + } + + return $this->weights; + } + + /** + * Returns the calculated lengths for included items. + * + * @return float[] + */ + public function get_item_lengths() { + if ( is_null( $this->lengths ) ) { + $this->lengths = array(); + + foreach ( $this->get_items() as $item ) { + $this->lengths[ $item->get_id() ] = $item->get_length() === '' ? 0 : $item->get_length(); + } + + if ( empty( $this->lengths ) ) { + $this->lengths = array( 0 ); + } + } + + return $this->lengths; + } + + public function get_item_volumes() { + if ( is_null( $this->volumes ) ) { + $this->volumes = array(); + + foreach ( $this->get_items() as $item ) { + $dimensions = $item->get_dimensions(); + $volume = ( '' !== $dimensions['length'] ? (float) $dimensions['length'] : 0 ) * ( '' !== $dimensions['width'] ? (float) $dimensions['width'] : 0 ) * ( '' !== $dimensions['height'] ? (float) $dimensions['height'] : 0 ); + + $this->volumes[ $item->get_id() ] = $volume * (float) $item->get_quantity(); + } + + if ( empty( $this->volumes ) ) { + $this->volumes = array( 0 ); + } + } + + return $this->volumes; + } + + /** + * Returns the calculated widths for included items. + * + * @return float[] + */ + public function get_item_widths() { + if ( is_null( $this->widths ) ) { + $this->widths = array(); + + foreach ( $this->get_items() as $item ) { + $this->widths[ $item->get_id() ] = $item->get_width() === '' ? 0 : $item->get_width(); + } + + if ( empty( $this->widths ) ) { + $this->widths = array( 0 ); + } + } + + return $this->widths; + } + + /** + * Returns the calculated heights for included items. + * + * @return float[] + */ + public function get_item_heights() { + if ( is_null( $this->heights ) ) { + $this->heights = array(); + + foreach ( $this->get_items() as $item ) { + $this->heights[ $item->get_id() ] = ( $item->get_height() === '' ? 0 : $item->get_height() ) * $item->get_quantity(); + } + + if ( empty( $this->heights ) ) { + $this->heights = array( 0 ); + } + } + + return $this->heights; + } + + /** + * Returns the calculated weight for included items. + * + * @return float + */ + public function get_content_weight() { + return wc_format_decimal( array_sum( $this->get_item_weights() ) ); + } + + public function get_content_dimensions() { + return array( + 'length' => $this->get_content_length(), + 'width' => $this->get_content_width(), + 'height' => $this->get_content_height(), + ); + } + + /** + * Returns the calculated length for included items. + * + * @return float + */ + public function get_content_length() { + $default = max( $this->get_item_lengths() ); + + return wc_format_decimal( $default, false, true ); + } + + /** + * Returns the calculated width for included items. + * + * @return float + */ + public function get_content_width() { + $default = max( $this->get_item_widths() ); + + return wc_format_decimal( $default, false, true ); + } + + /** + * Returns the calculated volume for included items. + * + * @return float + */ + public function get_content_volume() { + $default = array_sum( $this->get_item_volumes() ); + + return wc_format_decimal( $default, false, true ); + } + + /** + * Returns the calculated height for included items. + * + * @return float + */ + public function get_content_height() { + $default_height = array_sum( $this->get_item_heights() ); + + return wc_format_decimal( $default_height, false, true ); + } + + /** + * Returns the shipping address properties. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string[] + */ + public function get_address( $context = 'view' ) { + return $this->get_prop( 'address', $context ); + } + + /** + * Returns the shipment total. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return float + */ + public function get_total( $context = 'view' ) { + return $this->get_prop( 'total', $context ); + } + + /** + * Returns the shipment total. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return float + */ + public function get_subtotal( $context = 'view' ) { + $subtotal = $this->get_prop( 'subtotal', $context ); + + if ( 'view' === $context && empty( $subtotal ) ) { + $subtotal = $this->get_total(); + } + + return $subtotal; + } + + /** + * Returns the additional total amount containing shipping and fee costs. + * Only one of the shipments related to an order should include additional total. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return float + */ + public function get_additional_total( $context = 'view' ) { + return $this->get_prop( 'additional_total', $context ); + } + + public function has_tracking() { + $has_tracking = true; + + if ( ! $this->has_tracking_instruction() && ! $this->get_tracking_url() ) { + $has_tracking = false; + } + + /** + * Check whether the label supports tracking or not + */ + if ( $this->has_label() && ( $label = $this->get_label() ) ) { + if ( ! $label->is_trackable() ) { + $has_tracking = false; + } + } + + return apply_filters( "{$this->get_general_hook_prefix()}has_tracking", $has_tracking, $this ); + } + + /** + * Returns the shipment tracking id. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_tracking_id( $context = 'view' ) { + return $this->get_prop( 'tracking_id', $context ); + } + + /** + * Returns the shipment tracking URL. + * + * @return string + */ + public function get_tracking_url() { + $tracking_url = ''; + + if ( $provider = $this->get_shipping_provider_instance() ) { + $tracking_url = $provider->get_tracking_url( $this ); + } + + /** + * Filter to adjust a Shipment's tracking URL. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_tracking_url + * + * @param string $tracking_url The tracking URL. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}tracking_url", $tracking_url, $this ); + } + + /** + * Returns the shipment tracking instruction. + * + * @return string + */ + public function get_tracking_instruction( $plain = false ) { + $instruction = ''; + + if ( $provider = $this->get_shipping_provider_instance() ) { + $instruction = $provider->get_tracking_desc( $this, $plain ); + } + + /** + * Filter to adjust a Shipment's tracking instruction. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_tracking_instruction + * + * @param string $instruction The tracking instruction. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}tracking_instruction", $instruction, $this ); + } + + /** + * Returns whether the current shipment has tracking instructions available or not. + * + * @return boolean + */ + public function has_tracking_instruction() { + $instruction = $this->get_tracking_instruction( true ); + + return ( ! empty( $instruction ) ) ? true : false; + } + + /** + * Returns the shipment shipping provider. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_shipping_provider( $context = 'view' ) { + return $this->get_prop( 'shipping_provider', $context ); + } + + public function get_shipping_provider_title() { + if ( $provider = $this->get_shipping_provider_instance() ) { + return $provider->get_title(); + } + + return ''; + } + + public function get_shipping_provider_instance() { + $provider = $this->get_shipping_provider(); + + if ( ! empty( $provider ) ) { + return wc_gzd_get_shipping_provider( $provider ); + } + + return false; + } + + /** + * Returns the formatted shipping address. + * + * @param string $empty_content Content to show if no address is present. + * @return string + */ + public function get_formatted_address( $empty_content = '' ) { + $address = WC()->countries->get_formatted_address( $this->get_address() ); + + return $address ? $address : $empty_content; + } + + /** + * Get a formatted shipping address for the order. + * + * @return string + */ + public function get_address_map_url( $address ) { + // Remove name and company before generate the Google Maps URL. + unset( $address['first_name'], $address['last_name'], $address['company'], $address['email'], $address['phone'], $address['title'] ); + + /** + * Filter to adjust a Shipment's address parts used for constructing the Google maps URL. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_address_map_url_parts + * + * @param string[] $address The address parts used. + * @param Shipment $this The shipment object. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + $address = apply_filters( "{$this->get_hook_prefix()}address_map_url_parts", $address, $this ); + $address = array_filter( $address ); + + /** + * Filter to adjust a Shipment's address Google maps URL. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_address_map_url + * + * @param string $url The address url. + * @param Shipment $this The shipment object. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}address_map_url", 'https://maps.google.com/maps?&q=' . rawurlencode( implode( ', ', $address ) ) . '&z=16', $this ); + } + + /** + * Returns the shipment address phone number. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_phone( $context = 'view' ) { + return $this->get_address_prop( 'phone', $context ); + } + + /** + * Returns the shipment address email. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_email( $context = 'view' ) { + return $this->get_address_prop( 'email', $context ); + } + + /** + * Returns the shipment address first line. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_address_1( $context = 'view' ) { + return $this->get_address_prop( 'address_1', $context ); + } + + /** + * Returns the shipment address second line. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_address_2( $context = 'view' ) { + return $this->get_address_prop( 'address_2', $context ); + } + + /** + * Returns the shipment address street number by splitting the address. + * + * @param string $type The address type e.g. address_1 or address_2. + * + * @return string + */ + public function get_address_street_number( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_$type"}() ); + + /** + * Filter to adjust the shipment address street number. + * + * @param string $number The shipment address street number. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_get_shipment_address_street_number', $split['number'], $this ); + } + + /** + * Returns the shipment address street without number by splitting the address. + * + * @param string $type The address type e.g. address_1 or address_2. + * + * @return string + */ + public function get_address_street( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_$type"}() ); + + /** + * Filter to adjust the shipment address street. + * + * @param string $street The shipment address street without street number. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_get_shipment_address_street', $split['street'], $this ); + } + + public function get_address_street_addition( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_$type"}() ); + + /** + * Filter to adjust the shipment address street addition. + * + * @param string $addition The shipment address street addition e.g. EG14. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_get_shipment_address_street_addition', $split['addition'], $this ); + } + + public function get_address_street_addition_2( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_$type"}() ); + + /** + * Filter to adjust the shipment address street addition. + * + * @param string $addition The shipment address street addition e.g. EG14. + * @param Shipment $shipment The shipment object. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_get_shipment_address_street_addition_2', $split['addition_2'], $this ); + } + + /** + * Returns the shipment address company. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_company( $context = 'view' ) { + return $this->get_address_prop( 'company', $context ); + } + + /** + * Returns the shipment address first name. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_first_name( $context = 'view' ) { + return $this->get_address_prop( 'first_name', $context ); + } + + /** + * Returns the shipment address last name. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_last_name( $context = 'view' ) { + return $this->get_address_prop( 'last_name', $context ); + } + + /** + * Returns the shipment address formatted full name. + * + * @return string + */ + public function get_formatted_full_name() { + return sprintf( _x( '%1$s %2$s', 'full name', 'woocommerce-germanized' ), $this->get_first_name(), $this->get_last_name() ); + } + + /** + * Returns the shipment address postcode. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_postcode( $context = 'view' ) { + return $this->get_address_prop( 'postcode', $context ); + } + + /** + * Returns the shipment address city. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_city( $context = 'view' ) { + return $this->get_address_prop( 'city', $context ); + } + + /** + * Returns the shipment address state. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_state( $context = 'view' ) { + return $this->get_address_prop( 'state', $context ); + } + + public function get_formatted_state() { + if ( '' === $this->get_state() || '' === $this->get_country() ) { + return ''; + } + + return wc_gzd_get_formatted_state( $this->get_state(), $this->get_country() ); + } + + /** + * Returns the shipment address country. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_country( $context = 'view' ) { + return $this->get_address_prop( 'country', $context ) ? $this->get_address_prop( 'country', $context ) : ''; + } + + /** + * Returns the shipment address customs reference number. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_customs_reference_number( $context = 'view' ) { + return $this->get_address_prop( 'customs_reference_number', $context ) ? $this->get_address_prop( 'customs_reference_number', $context ) : ''; + } + + /** + * Returns a sender address prop by checking the corresponding provider and falling back to + * global sender address setting data. + * + * @param string $prop + * @param string $context + * + * @return null|string + */ + protected function get_sender_address_prop( $prop, $context = 'view' ) { + $value = null; + + if ( $provider = $this->get_shipping_provider_instance() ) { + $getter = "get_shipper_{$prop}"; + + if ( is_callable( array( $provider, $getter ) ) ) { + $value = $provider->$getter( $context ); + } + } else { + $key = "woocommerce_gzd_shipments_shipper_address_{$prop}"; + $value = get_option( $key, '' ); + } + + if ( 'view' === $context ) { + /** + * Filter to adjust a shipment's sender address property e.g. first_name. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. `$prop` refers to the actual address property e.g. first_name. + * + * Example hook name: woocommerce_gzd_shipment_get_sender_address_first_name + * + * @param string $value The address property value. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $value = apply_filters( "{$this->get_hook_prefix()}sender_address_{$prop}", $value, $this ); + } + + return $value; + } + + /** + * Returns the formatted sender address. + * + * @param string $empty_content Content to show if no address is present. + * @return string + */ + public function get_formatted_sender_address( $empty_content = '' ) { + $address = WC()->countries->get_formatted_address( $this->get_sender_address() ); + + return $address ? $address : $empty_content; + } + + /** + * Returns the address of the sender e.g. customer. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string[] + */ + public function get_sender_address( $context = 'view' ) { + return apply_filters( + "{$this->get_hook_prefix()}sender_address", + array( + 'company' => $this->get_sender_company( $context ), + 'first_name' => $this->get_sender_first_name( $context ), + 'last_name' => $this->get_sender_last_name( $context ), + 'address_1' => $this->get_sender_address_1( $context ), + 'address_2' => $this->get_sender_address_2( $context ), + 'postcode' => $this->get_sender_postcode( $context ), + 'city' => $this->get_sender_city( $context ), + 'country' => $this->get_sender_country( $context ), + 'state' => $this->get_sender_state( $context ), + ), + $this + ); + } + + /** + * Returns the sender address phone number. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_phone( $context = 'view' ) { + return $this->get_sender_address_prop( 'phone', $context ); + } + + /** + * Returns the sender address email. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_email( $context = 'view' ) { + return $this->get_sender_address_prop( 'email', $context ); + } + + /** + * Returns the sender address first line. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_address_1( $context = 'view' ) { + return $this->get_sender_address_prop( 'address_1', $context ); + } + + /** + * Returns the sender address second line. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_address_2( $context = 'view' ) { + return $this->get_sender_address_prop( 'address_2', $context ); + } + + /** + * Returns the sender address street number by splitting the address. + * + * @param string $type The address type e.g. address_1 or address_2. + * + * @return string + */ + public function get_sender_address_street_number( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_sender_$type"}() ); + + return $split['number']; + } + + /** + * Returns the sender address street without number by splitting the address. + * + * @param string $type The address type e.g. address_1 or address_2. + * + * @return string + */ + public function get_sender_address_street( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_sender_$type"}() ); + + return $split['street']; + } + + /** + * Returns the sender address street addition by splitting the address. + * + * @param string $type The address type e.g. address_1 or address_2. + * + * @return string + */ + public function get_sender_address_street_addition( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_sender_$type"}() ); + + return $split['addition']; + } + + public function get_sender_address_street_addition_2( $type = 'address_1' ) { + $split = wc_gzd_split_shipment_street( $this->{"get_sender_$type"}() ); + + return $split['addition_2']; + } + + /** + * Returns the sender address company. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_company( $context = 'view' ) { + return $this->get_sender_address_prop( 'company', $context ); + } + + /** + * Returns the sender address first name. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_first_name( $context = 'view' ) { + return $this->get_sender_address_prop( 'first_name', $context ); + } + + /** + * Returns the shipment address last name. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_last_name( $context = 'view' ) { + return $this->get_sender_address_prop( 'last_name', $context ); + } + + /** + * Returns the sender address formatted full name. + * + * @return string + */ + public function get_formatted_sender_full_name() { + return sprintf( _x( '%1$s %2$s', 'full name', 'woocommerce-germanized' ), $this->get_sender_first_name(), $this->get_sender_last_name() ); + } + + /** + * Returns the sender address postcode. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_postcode( $context = 'view' ) { + return $this->get_sender_address_prop( 'postcode', $context ); + } + + /** + * Returns the sender address city. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_city( $context = 'view' ) { + return $this->get_sender_address_prop( 'city', $context ); + } + + /** + * Returns the sender address state. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_state( $context = 'view' ) { + return $this->get_sender_address_prop( 'state', $context ); + } + + /** + * Returns the sender address customs reference number. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_customs_reference_number( $context = 'view' ) { + return $this->get_sender_address_prop( 'customs_reference_number', $context ) ? $this->get_sender_address_prop( 'customs_reference_number', $context ) : ''; + } + + /** + * Returns the sender address customs reference number. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_customs_uk_vat_id( $context = 'view' ) { + return $this->get_sender_address_prop( 'customs_uk_vat_id', $context ) ? $this->get_sender_address_prop( 'customs_uk_vat_id', $context ) : ''; + } + + public function get_formatted_sender_state() { + if ( '' === $this->get_sender_state() || '' === $this->get_sender_country() ) { + return ''; + } + + return wc_gzd_get_formatted_state( $this->get_sender_state(), $this->get_sender_country() ); + } + + /** + * Returns the sender address country. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return string + */ + public function get_sender_country( $context = 'view' ) { + return $this->get_sender_address_prop( 'country', $context ) ? $this->get_sender_address_prop( 'country', $context ) : ''; + } + + /** + * Return the date this shipment is estimated to be delivered. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return WC_DateTime|null object if the date is set or null if there is no date. + */ + public function get_est_delivery_date( $context = 'view' ) { + return $this->get_prop( 'est_delivery_date', $context ); + } + + /** + * Decides whether the shipment is sent to an external pickup or not. + * + * @param string[]|string $types + * + * @return boolean + */ + public function send_to_external_pickup( $types = array() ) { + $types = is_array( $types ) ? $types : array( $types ); + + /** + * Filter to decide whether a Shipment is to be sent to a external pickup location + * e.g. packstation. + * + * @param boolean $external True if the Shipment goes to a pickup location. + * @param array $types Array containing the types to be checked against, or empty. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_send_to_external_pickup', false, $types, $this ); + } + + /** + * Returns an address prop. + * + * @param string $prop + * @param string $context + * + * @return null|string + */ + protected function get_address_prop( $prop, $context = 'view' ) { + $value = null; + + if ( isset( $this->changes['address'][ $prop ] ) || isset( $this->data['address'][ $prop ] ) ) { + $value = isset( $this->changes['address'][ $prop ] ) ? $this->changes['address'][ $prop ] : $this->data['address'][ $prop ]; + + if ( 'view' === $context ) { + /** + * Filter to adjust a Shipment's shipping address property e.g. first_name. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. `$prop` refers to the actual address property e.g. first_name. + * + * Example hook name: woocommerce_gzd_shipment_get_address_first_name + * + * @param string $value The address property value. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $value = apply_filters( "{$this->get_hook_prefix()}address_{$prop}", $value, $this ); + } + } + + return $value; + } + + /** + * Returns dimensions. + * + * @return string|array + */ + public function get_dimensions( $context = 'view' ) { + return array( + 'length' => $this->get_length( $context ), + 'width' => $this->get_width( $context ), + 'height' => $this->get_height( $context ), + ); + } + + /** + * Returns dimensions. + * + * @return string|array + */ + public function get_package_dimensions() { + return array( + 'length' => $this->get_package_length(), + 'width' => $this->get_package_width(), + 'height' => $this->get_package_height(), + ); + } + + public function get_formatted_dimensions() { + return wc_gzd_format_shipment_dimensions( $this->get_dimensions(), $this->get_dimension_unit() ); + } + + /** + * Returns whether the shipment is editable or not. + * + * @return boolean + */ + public function is_editable() { + /** + * Filter to dedice whether the current Shipment is still editable or not. + * + * @param boolean $is_editable Whether the Shipment is editable or not. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_is_editable', $this->has_status( wc_gzd_get_shipment_editable_statuses() ), $this ); + } + + /** + * Returns the shipment number. + * + * @return string + */ + public function get_shipment_number() { + /** + * Filter to adjust a Shipment's number. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_shipment_number + * + * @param string $number The shipment number. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return (string) apply_filters( "{$this->get_hook_prefix()}shipment_number", $this->get_id(), $this ); + } + + /* + |-------------------------------------------------------------------------- + | Setters + |-------------------------------------------------------------------------- + */ + + /** + * Set shipment status. + * + * @param string $new_status Status to change the shipment to. No internal gzd- prefix is required. + * @param boolean $manual_update Whether it is a manual status update or not. + * @return array details of change + */ + public function set_status( $new_status, $manual_update = false ) { + $old_status = $this->get_status(); + $new_status = 'gzd-' === substr( $new_status, 0, 4 ) ? substr( $new_status, 4 ) : $new_status; + + $this->set_prop( 'status', $new_status ); + + $result = array( + 'from' => $old_status, + 'to' => $new_status, + ); + + if ( true === $this->object_read && ! empty( $result['from'] ) && $result['from'] !== $result['to'] ) { + $this->status_transition = array( + 'from' => ! empty( $this->status_transition['from'] ) ? $this->status_transition['from'] : $result['from'], + 'to' => $result['to'], + 'manual' => (bool) $manual_update, + ); + + if ( $manual_update ) { + /** + * Action that fires after a shipment status has been updated manually. + * + * @param integer $shipment_id The shipment id. + * @param string $status The new shipment status. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_edit_status', $this->get_id(), $result['to'] ); + } + + $this->maybe_set_date_sent(); + } + + return $result; + } + + public function is_shipped() { + $is_shipped = $this->has_status( wc_gzd_get_shipment_sent_statuses() ); + + return apply_filters( $this->get_hook_prefix() . 'is_shipped', $is_shipped, $this ); + } + + /** + * Maybe set date sent. + * + * Sets the date sent variable when transitioning to the shipped shipment status. + * Date sent is set once in this manner - only when it is not already set. + */ + public function maybe_set_date_sent() { + // This logic only runs if the date_sent prop has not been set yet. + if ( ! $this->get_date_sent( 'edit' ) ) { + if ( $this->is_shipped() ) { + // If payment complete status is reached, set paid now. + $this->set_date_sent( time() ); + } + } + } + + /** + * Updates status of shipment immediately. + * + * @uses Shipment::set_status() + * + * @param string $new_status Status to change the shipment to. No internal gzd- prefix is required. + * @param bool $manual Is this a manual order status change? + * @return bool + */ + public function update_status( $new_status, $manual = false ) { + if ( ! $this->get_id() ) { + return false; + } + + try { + $this->set_status( $new_status, $manual ); + $this->save(); + } catch ( Exception $e ) { + $logger = wc_get_logger(); + $logger->error( + sprintf( 'Error updating status for shipment #%d', $this->get_id() ), + array( + 'shipment' => $this, + 'error' => $e, + ) + ); + return false; + } + return true; + } + + /** + * Set the date this shipment was created. + * + * @param string|integer|null $date UTC timestamp, or ISO 8601 DateTime. If the DateTime string has no timezone or offset, WordPress site timezone will be assumed. Null if their is no date. + */ + public function set_date_created( $date = null ) { + $this->set_date_prop( 'date_created', $date ); + } + + /** + * Set the date this shipment was sent. + * + * @param string|integer|null $date UTC timestamp, or ISO 8601 DateTime. If the DateTime string has no timezone or offset, WordPress site timezone will be assumed. Null if their is no date. + */ + public function set_date_sent( $date = null ) { + $this->set_date_prop( 'date_sent', $date ); + } + + /** + * Set shipment weight in kg. + * + * @param string $weight The weight. + */ + public function set_weight( $weight ) { + $this->set_prop( 'weight', '' === $weight ? '' : wc_format_decimal( $weight ) ); + } + + public function set_packaging_weight( $weight ) { + $this->set_prop( 'packaging_weight', '' === $weight ? '' : wc_format_decimal( $weight ) ); + } + + /** + * Set shipment total weight. + * + * @param string $weight The weight. + */ + public function set_total_weight( $weight ) { + $this->set_prop( 'total_weight', '' === $weight ? '' : wc_format_decimal( $weight ) ); + } + + /** + * Set shipment width. + * + * @param string $width The width. + */ + public function set_width( $width ) { + $this->set_prop( 'width', '' === $width ? '' : wc_format_decimal( $width ) ); + } + + public function set_weight_unit( $unit ) { + $this->set_prop( 'weight_unit', $unit ); + } + + public function set_dimension_unit( $unit ) { + $this->set_prop( 'dimension_unit', $unit ); + } + + /** + * Set shipment length. + * + * @param string $length The length. + */ + public function set_length( $length ) { + $this->set_prop( 'length', '' === $length ? '' : wc_format_decimal( $length ) ); + } + + /** + * Set shipment height. + * + * @param string $height The height. + */ + public function set_height( $height ) { + $this->set_prop( 'height', '' === $height ? '' : wc_format_decimal( $height ) ); + } + + /** + * Set shipment address. + * + * @param string[] $address The address props. + */ + public function set_address( $address ) { + $this->set_prop( 'address', empty( $address ) ? array() : (array) $address ); + } + + /** + * Set shipment shipping method. + * + * @param string $method The shipping method. + */ + public function set_shipping_method( $method ) { + $this->shipping_method_instance = null; + + $this->set_prop( 'shipping_method', $method ); + } + + /** + * Set shipment version. + * + * @param string $version The version. + */ + public function set_version( $version ) { + $this->set_prop( 'version', $version ); + } + + /** + * Set the date this shipment will be delivered. + * + * @param string|integer|null $date UTC timestamp, or ISO 8601 DateTime. If the DateTime string has no timezone or offset, WordPress site timezone will be assumed. Null if their is no date. + */ + public function set_est_delivery_date( $date = null ) { + $this->set_date_prop( 'est_delivery_date', $date ); + } + + /** + * Set shipment total. + * + * @param float|string $value The shipment total. + */ + public function set_total( $value ) { + $value = wc_format_decimal( $value ); + + if ( ! is_numeric( $value ) ) { + $value = 0; + } + + $this->set_prop( 'total', $value ); + } + + /** + * Set shipment total. + * + * @param float|string $value The shipment total. + */ + public function set_subtotal( $value ) { + $value = wc_format_decimal( $value ); + + if ( ! is_numeric( $value ) ) { + $value = 0; + } + + $this->set_prop( 'subtotal', $value ); + } + + /** + * Set shipment additional total. + * + * @param float|string $value The shipment total. + */ + public function set_additional_total( $value ) { + $value = wc_format_decimal( $value ); + + if ( ! is_numeric( $value ) ) { + $value = 0; + } + + $this->set_prop( 'additional_total', $value ); + } + + /** + * Set shipment shipping country. + * + * @param string $country The country in ISO format. + */ + public function set_country( $country ) { + $this->set_address_prop( 'country', $country ); + } + + /** + * Update a specific address prop. + * + * @param $prop + * @param $value + */ + protected function set_address_prop( $prop, $value ) { + $address = $this->get_address(); + $address[ $prop ] = $value; + + $this->set_address( $address ); + } + + /** + * Set shipment tracking id. + * + * @param string $tracking_id The trakcing id. + */ + public function set_tracking_id( $tracking_id ) { + $this->set_prop( 'tracking_id', $tracking_id ); + } + + /** + * Set shipment shipping provider. + * + * @param string $provider The shipping provider. + */ + public function set_shipping_provider( $provider ) { + $this->set_prop( 'shipping_provider', wc_gzd_get_shipping_provider_slug( $provider ) ); + } + + /** + * Set packaging id. + * + * @param integer $packaging_id The packaging id. + */ + public function set_packaging_id( $packaging_id ) { + $this->set_prop( 'packaging_id', absint( $packaging_id ) ); + + $this->packaging = null; + } + + public function sync_packaging() { + $available_packaging = $this->get_selectable_packaging(); + $default_packaging = $this->get_default_packaging(); + $packaging_id = $this->get_packaging_id( 'edit' ); + + if ( ! empty( $packaging_id ) ) { + $exists = false; + + foreach ( $available_packaging as $packaging ) { + if ( (int) $packaging_id === (int) $packaging->get_id() ) { + $exists = true; + break; + } + } + + if ( ! $exists && $default_packaging ) { + $this->set_packaging_id( $default_packaging->get_id() ); + } + } elseif ( empty( $packaging_id ) && $default_packaging ) { + $this->set_packaging_id( $default_packaging->get_id() ); + } + } + + public function update_packaging() { + if ( $packaging = $this->get_packaging() ) { + $packaging_dimension = wc_gzd_get_packaging_dimension_unit(); + + $props = array( + 'width' => wc_get_dimension( $packaging->get_width( 'edit' ), $this->get_dimension_unit(), $packaging_dimension ), + 'length' => wc_get_dimension( $packaging->get_length( 'edit' ), $this->get_dimension_unit(), $packaging_dimension ), + 'height' => wc_get_dimension( $packaging->get_height( 'edit' ), $this->get_dimension_unit(), $packaging_dimension ), + 'packaging_weight' => wc_get_weight( $packaging->get_weight( 'edit' ), $this->get_weight_unit(), wc_gzd_get_packaging_weight_unit() ), + ); + + $this->set_props( $props ); + } else { + $props = array( 'packaging_weight' => '' ); + $changes = $this->get_changes(); + + /** + * Maybe reset dimensions in case they've not been explicitly set + */ + if ( array_key_exists( 'packaging_id', $changes ) ) { + foreach ( array( 'length', 'width', 'height' ) as $dim_prop ) { + if ( ! array_key_exists( $dim_prop, $changes ) ) { + $props = array_merge( $props, array( $dim_prop => '' ) ); + } + } + } + + // Reset + $this->set_props( $props ); + } + + return true; + } + + /** + * Return an array of items within this shipment. + * + * @return ShipmentItem[] + */ + public function get_items() { + $items = array(); + + if ( is_null( $this->items ) ) { + $this->items = array_filter( $this->data_store->read_items( $this ) ); + + $items = (array) $this->items; + } else { + $items = (array) $this->items; + } + + /** + * Filter to adjust items belonging to a Shipment. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_items + * + * @param string $number The shipment number. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}items", $items, $this ); + } + + /** + * Get's the URL to edit the shipment in the backend. + * + * @return string + */ + abstract public function get_edit_shipment_url(); + + public function get_view_shipment_url() { + /** + * Filter to adjust the URL being used to access the view shipment page on the customer account page. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_view_shipment_url + * + * @param string $url The URL pointing to the view page. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}_view_shipment_url", wc_get_endpoint_url( 'view-shipment', $this->get_id(), wc_get_page_permalink( 'myaccount' ) ), $this ); + } + + /** + * Get an item object. + * + * @param int $item_id ID of item to get. + * + * @return ShipmentItem|false + */ + public function get_item( $item_id ) { + $items = $this->get_items(); + + if ( isset( $items[ $item_id ] ) ) { + return $items[ $item_id ]; + } + + return false; + } + + /** + * Remove item from the shipment. + * + * @param int $item_id Item ID to delete. + * + * @return false|void + */ + public function remove_item( $item_id ) { + $item = $this->get_item( $item_id ); + + // Unset and remove later. + $this->items_to_delete[] = $item; + + unset( $this->items[ $item->get_id() ] ); + + $this->reset_content_data(); + $this->calculate_totals(); + $this->sync_packaging(); + } + + public function update_item_quantity( $item_id, $quantity = 1 ) { + if ( $item = $this->get_item( $item_id ) ) { + $item->set_quantity( $quantity ); + + if ( array_key_exists( 'quantity', $item->get_changes() ) ) { + $this->sync_packaging(); + } + + return true; + } + + return false; + } + + /** + * Adds a shipment item to this shipment. The shipment item will not persist until save. + * + * @since 3.0.0 + * @param ShipmentItem $item Shipment item object. + * + * @return false|void + */ + public function add_item( $item ) { + // Make sure that items are loaded + $items = $this->get_items(); + + // Set parent. + $item->set_shipment_id( $this->get_id() ); + + // Append new row with generated temporary ID. + $item_id = $item->get_id(); + + if ( $item_id ) { + $this->items[ $item_id ] = $item; + } else { + $this->items[ 'new:' . count( $this->items ) ] = $item; + } + + $this->items_to_pack = null; + + $this->reset_content_data(); + $this->calculate_totals(); + $this->sync_packaging(); + } + + /** + * Reset item content data. + */ + protected function reset_content_data() { + $this->weights = null; + $this->lengths = null; + $this->widths = null; + $this->heights = null; + $this->volumes = null; + } + + /** + * Handle the status transition. + */ + protected function status_transition() { + $status_transition = $this->status_transition; + + // Reset status transition variable. + $this->status_transition = false; + + if ( $status_transition ) { + try { + /** + * Action that fires before a shipment status transition happens. + * + * @param integer $shipment_id The shipment id. + * @param Shipment $shipment The shipment object. + * @param array $status_transition The status transition data. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_before_status_change', $this->get_id(), $this, $this->status_transition ); + + $status_to = $status_transition['to']; + $status_hook_prefix = 'woocommerce_gzd_' . ( 'simple' === $this->get_type() ? '' : $this->get_type() . '_' ) . 'shipment_status'; + + /** + * Action that indicates shipment status change to a specific status. + * + * The dynamic portion of the hook name, `$status_hook_prefix` constructs a unique prefix + * based on the shipment type. `$status_to` refers to the new shipment status. + * + * Example hook name: `woocommerce_gzd_return_shipment_status_processing` + * + * @param integer $shipment_id The shipment id. + * @param Shipment $shipment The shipment object. + * + * @see wc_gzd_get_shipment_statuses() + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$status_hook_prefix}_$status_to", $this->get_id(), $this ); + + if ( ! empty( $status_transition['from'] ) ) { + $status_from = $status_transition['from']; + + /** + * Action that indicates shipment status change from a specific status to a specific status. + * + * The dynamic portion of the hook name, `$status_hook_prefix` constructs a unique prefix + * based on the shipment type. `$status_from` refers to the old shipment status. + * `$status_to` refers to the new status. + * + * Example hook name: `woocommerce_gzd_return_shipment_status_processing_to_shipped` + * + * @param integer $shipment_id The shipment id. + * @param Shipment $shipment The shipment object. + * + * @see wc_gzd_get_shipment_statuses() + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$status_hook_prefix}_{$status_from}_to_{$status_to}", $this->get_id(), $this ); + + /** + * Action that indicates shipment status change. + * + * @param integer $shipment_id The shipment id. + * @param string $status_from The old shipment status. + * @param string $status_to The new shipment status. + * @param Shipment $shipment The shipment object. + * + * @see wc_gzd_get_shipment_statuses() + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_status_changed', $this->get_id(), $status_from, $status_to, $this ); + } + } catch ( Exception $e ) { + $logger = wc_get_logger(); + $logger->error( + sprintf( 'Status transition of shipment #%d errored!', $this->get_id() ), + array( + 'shipment' => $this, + 'error' => $e, + ) + ); + } + } + } + + /** + * Remove all items from the shipment. + */ + public function remove_items() { + $this->data_store->delete_items( $this ); + $this->items = array(); + + $this->items_to_pack = null; + + $this->reset_content_data(); + $this->calculate_totals(); + $this->sync_packaging(); + } + + /** + * Save all items which are part of this shipment. + */ + protected function save_items() { + $items_changed = false; + + foreach ( $this->items_to_delete as $item ) { + $item->delete(); + $items_changed = true; + } + + $this->items_to_delete = array(); + + foreach ( $this->get_items() as $item_key => $item ) { + $item->set_shipment_id( $this->get_id() ); + + $item_id = $item->save(); + + // If ID changed (new item saved to DB)... + if ( $item_id !== $item_key ) { + $this->items[ $item_id ] = $item; + + unset( $this->items[ $item_key ] ); + + $items_changed = true; + } + } + } + + /** + * Finds an ShipmentItem based on an order item id. + * + * @param integer $order_item_id + * + * @return bool|ShipmentItem + */ + public function get_item_by_order_item_id( $order_item_id ) { + $items = $this->get_items(); + + foreach ( $items as $item ) { + if ( $item->get_order_item_id() === (int) $order_item_id ) { + return $item; + } + } + + return false; + } + + /** + * Returns version. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return integer + */ + public function get_version( $context = 'view' ) { + return $this->get_prop( 'version', $context ); + } + + /** + * Returns the packaging id belonging to the shipment. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return integer + */ + public function get_packaging_id( $context = 'view' ) { + return $this->get_prop( 'packaging_id', $context ); + } + + public function get_packaging() { + if ( is_null( $this->packaging ) && $this->get_packaging_id() > 0 ) { + if ( $packaging = wc_gzd_get_packaging( $this->get_packaging_id() ) ) { + // Do only allow load packaging if it does really exist in DB. + if ( $packaging->get_id() > 0 ) { + $this->packaging = $packaging; + } + } + } + + return $this->packaging; + } + + /** + * Returns a list of available (fitting) packaging options for the current shipment + * + * @return Packaging[] + */ + public function get_available_packaging() { + $packaging_store = \WC_Data_Store::load( 'packaging' ); + + return apply_filters( "{$this->get_hook_prefix()}available_packaging", $packaging_store->find_available_packaging_for_shipment( $this ), $this ); + } + + /** + * Returns a list of user-selectable packaging options for the current shipment. + * + * @return Packaging[] + */ + public function get_selectable_packaging() { + return apply_filters( "{$this->get_hook_prefix()}selectable_packaging", $this->get_available_packaging(), $this ); + } + + public function get_default_packaging() { + $packaging_store = \WC_Data_Store::load( 'packaging' ); + $default_packaging = $packaging_store->find_best_match_for_shipment( $this ); + + if ( ! $default_packaging ) { + $setting = Package::get_setting( 'default_packaging' ); + + if ( ! empty( $setting ) && wc_gzd_get_packaging( $setting ) ) { + $default_packaging = wc_gzd_get_packaging( $setting ); + } + } + + return apply_filters( "{$this->get_hook_prefix()}default_packaging_id", $default_packaging, $this ); + } + + /** + * Tries to fetch the order for the current shipment. + * + * @return bool|WC_Order|null + */ + abstract public function get_order(); + + abstract public function get_order_id(); + + /** + * Returns the formatted order number. + * + * @return string + */ + public function get_order_number() { + if ( $order = $this->get_order() ) { + return $order->get_order_number(); + } + + return $this->get_order_id(); + } + + /** + * Returns whether the Shipment contains an order item or not. + * + * @param integer|integer[] $item_id + * + * @return boolean + */ + public function contains_order_item( $item_id ) { + + if ( ! is_array( $item_id ) ) { + $item_id = array( $item_id ); + } + + $new_items = $item_id; + + foreach ( $item_id as $key => $order_item_id ) { + + if ( is_a( $order_item_id, 'WC_Order_Item' ) ) { + $order_item_id = $order_item_id->get_id(); + $item_id[ $key ] = $order_item_id; + } + + if ( $this->get_item_by_order_item_id( $order_item_id ) ) { + unset( $new_items[ $key ] ); + } + } + + $contains = empty( $new_items ) ? true : false; + + /** + * Filter to adjust whether a Shipment contains a specific order item or not. + * + * @param boolean $contains Whether the Shipment contains the order item or not. + * @param integer[] $order_item_id The order item id(s). + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_contains_order_item', $contains, $item_id, $this ); + } + + public function get_shippable_item_count() { + return 0; + } + + /** + * Finds an ShipmentItem based on an item parent id. + * + * @param integer $item_parent_id + * + * @return bool|ShipmentItem + */ + public function get_item_by_item_parent_id( $item_parent_id ) { + $items = $this->get_items(); + + foreach ( $items as $item ) { + if ( $item->get_parent_id() === $item_parent_id ) { + return $item; + } + } + + return false; + } + + public function needs_items( $available_items = false ) { + return false; + } + + public function sync( $args = array() ) { + return false; + } + + public function sync_items( $args = array() ) { + return false; + } + + /** + * Returns a label + * + * @return boolean|ShipmentLabel|ShipmentReturnLabel + */ + public function get_label() { + $label = false; + $prefix = ''; + + if ( $provider = $this->get_shipping_provider_instance() ) { + $prefix = $provider->get_name() . '_'; + $label = $provider->get_label( $this ); + } + + /** + * Filter for shipping providers to retrieve the `ShipmentLabel` corresponding to a certain shipment. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. `$provider` is related to the current shipping provider + * for the shipment (slug). + * + * Example hook name: `woocommerce_gzd_return_shipment_get_dhl_label` + * + * @param boolean|ShipmentLabel $label The label instance. + * @param Shipment $shipment The current shipment instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}{$prefix}label", $label, $this ); + } + + /** + * Output label admin fields. + */ + public function get_label_settings_html() { + $hook_prefix = $this->get_general_hook_prefix(); + $html = ''; + + if ( $provider = $this->get_shipping_provider_instance() ) { + $hook_prefix = $hook_prefix . '_' . $provider->get_name(); + $html = $provider->get_label_fields_html( $this ); + } + + /** + * Action for shipping providers to output available admin settings while creating a label. + * + * The dynamic portion of this hook, `$hook_prefix` is used to construct a + * unique hook for a shipment type. `$provider` is related to the current shipping provider + * for the shipment (slug). + * + * Example hook name: `woocommerce_gzd_return_shipment_print_dhl_label_admin_fields` + * + * @param Shipment $shipment The current shipment instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$hook_prefix}label_settings_html", $html, $this ); + } + + /** + * @param $props + * + * @return true|ShipmentError + */ + public function create_label( $props = false ) { + $hook_prefix = $this->get_general_hook_prefix(); + $provider_name = ''; + $error = new ShipmentError(); + + /** + * Sanitize props + */ + if ( is_array( $props ) ) { + foreach ( $props as $key => $value ) { + $props[ $key ] = wc_clean( wp_unslash( $value ) ); + } + } + + if ( $provider = $this->get_shipping_provider_instance() ) { + $provider_name = $provider->get_name(); + $result = $provider->create_label( $this, $props ); + + if ( is_wp_error( $result ) ) { + $error = wc_gzd_get_shipment_error( $result ); + + if ( ! $error->is_soft_error() ) { + return $error; + } + } + } else { + /** + * Action for shipping providers to create the `ShipmentLabel` corresponding to a certain shipment. + * + * The dynamic portion of this hook, `$hook_prefix` is used to construct a + * unique hook for a shipment type. `$provider` is related to the current shipping provider + * for the shipment (slug). + * + * Example hook name: `woocommerce_gzd_return_shipment_create_dhl_label` + * + * @param array|false $props Array containing props extracted from post data (if created manually). + * @param WP_Error $error An WP_Error instance useful for returning errors while creating the label. + * @param Shipment $shipment The current shipment instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$hook_prefix}create_{$provider_name}label", $props, $error, $this ); + + if ( wc_gzd_shipment_wp_error_has_errors( $error ) && ! $error->is_soft_error() ) { + return $error; + } + } + + if ( $label = $this->get_label() ) { + $this->set_tracking_id( $label->get_number() ); + + /** + * Action for shipping providers to adjust the shipment before updating it after a label has + * been successfully generated. + * + * The dynamic portion of this hook, `$hook_prefix` is used to construct a + * unique hook for a shipment type. `$provider` is related to the current shipping provider + * for the shipment (slug). + * + * Example hook name: `woocommerce_gzd_return_shipment_created_dhl_label` + * + * @param Shipment $shipment The current shipment instance. + * @param array $props Array containing props extracted from post data (if created manually) and sanitized via `wc_clean`. + * @param array $raw_data Raw post data unsanitized. + * + * @since 3.1.2 + * @package Vendidero/Germanized/Shipments + */ + do_action( "{$hook_prefix}created_{$provider_name}label", $this, $props ); + + do_action( "{$hook_prefix}created_label", $this, $props ); + + $this->save(); + } + + if ( wc_gzd_shipment_wp_error_has_errors( $error ) ) { + return $error; + } + + return true; + } + + public function delete_label( $force = false ) { + if ( $this->supports_label() && ( $label = $this->get_label() ) ) { + $label->delete( $force ); + $this->set_tracking_id( '' ); + $this->save(); + + return true; + } + + return false; + } + + /** + * Whether or not the current shipments supports labels or not. + * + * @return bool + */ + public function supports_label() { + if ( $provider = $this->get_shipping_provider_instance() ) { + if ( $provider->supports_labels( $this->get_type(), $this ) ) { + return true; + } + } + + return false; + } + + /** + * Whether or not the current shipments needs a label or not. + * + * @return bool + */ + public function needs_label( $check_status = true ) { + $needs_label = true; + $provider = $this->get_shipping_provider(); + $hook_prefix = $this->get_general_hook_prefix(); + + if ( ! empty( $provider ) ) { + $provider = $provider . '_'; + } + + if ( $this->has_label() ) { + $needs_label = false; + } + + if ( ! $this->supports_label() ) { + $needs_label = false; + } + + if ( $shipping_provider = $this->get_shipping_provider_instance() ) { + if ( ! $shipping_provider->is_activated() ) { + $needs_label = false; + } + } + + // If shipment is already delivered + if ( $check_status && $this->is_shipped() ) { + $needs_label = false; + } + + /** + * Filter for shipping providers to decide whether the shipment needs a label or not. + * + * The dynamic portion of this hook, `$hook_prefix` is used to construct a + * unique hook for a shipment type. `$provider` is related to the current shipping provider + * for the shipment (slug). + * + * Example hook name: `woocommerce_gzd_return_shipment_needs_dhl_label` + * + * @param boolean $needs_label Whether or not the shipment needs a label. + * @param boolean $check_status Whether or not checking the shipment status is needed. + * @param Shipment $shipment The current shipment instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$hook_prefix}needs_{$provider}label", $needs_label, $check_status, $this ); + } + + /** + * Whether or not the current shipment has a valid label or not. + * + * @return bool + */ + public function has_label() { + $label = $this->get_label(); + + if ( $label && is_a( $label, '\Vendidero\Germanized\Shipments\Interfaces\ShipmentLabel' ) ) { + if ( 'return' === $label->get_type() ) { + if ( ! is_a( $label, '\Vendidero\Germanized\Shipments\Interfaces\ShipmentReturnLabel' ) ) { + return false; + } + } + + return true; + } else { + return false; + } + } + + public function get_label_download_url( $args = array() ) { + $download_url = ''; + $provider = $this->get_shipping_provider(); + + if ( $label = $this->get_label() ) { + $download_url = $label->get_download_url( $args ); + } + + /** + * Filter for shipping providers to adjust the label download URL. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. `$provider` is related to the current shipping provider + * for the shipment (slug). + * + * Example hook name: `woocommerce_gzd_return_shipment_get_dhl_label_download_url` + * + * @param string $url The download URL. + * @param Shipment $shipment The current shipment instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}{$provider}label_download_url", $download_url, $this ); + } + + public function add_note( $note, $added_by_user = false ) { + if ( $order = $this->get_order() ) { + if ( is_callable( array( $order, 'add_order_note' ) ) ) { + $order->add_order_note( $note, 0, $added_by_user ); + } + } + } + + /** + * Calculate totals based on contained items. + */ + protected function calculate_totals() { + $total = 0; + $subtotal = 0; + + foreach ( $this->get_items() as $item ) { + $total += round( $item->get_total(), wc_get_price_decimals() ); + $subtotal += round( $item->get_subtotal(), wc_get_price_decimals() ); + } + + $this->set_total( $total ); + + if ( empty( $subtotal ) ) { + $subtotal = $total; + } + + $this->set_subtotal( $subtotal ); + } + + public function delete( $force_delete = false ) { + $this->delete_label( $force_delete ); + + return parent::delete( $force_delete ); + } + + /** + * Save data to the database. + * + * @return integer shipment id + */ + public function save() { + try { + $this->calculate_totals(); + $is_new = false; + + if ( array_key_exists( 'packaging_id', $this->get_changes() ) || $this->is_editable() ) { + $this->update_packaging(); + } + + if ( $this->data_store ) { + // Trigger action before saving to the DB. Allows you to adjust object props before save. + do_action( 'woocommerce_before_' . $this->object_type . '_object_save', $this, $this->data_store ); + + if ( $this->get_id() ) { + $this->data_store->update( $this ); + } else { + $this->data_store->create( $this ); + $is_new = true; + } + } + + $this->save_items(); + + /** + * Trigger action after saving shipment to the DB. + * + * @param Shipment $shipment The shipment object being saved. + * @param WC_Data_Store_WP $data_store THe data store persisting the data. + */ + do_action( 'woocommerce_after_' . $this->object_type . '_object_save', $this, $this->data_store ); + + $hook_postfix = ''; + + if ( 'simple' !== $this->get_type() ) { + $hook_postfix = $this->get_type() . '_'; + } + + /** + * Trigger action after saving shipment to the DB. + * + * The dynamic portion of this hook, `$hook_postfix` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_after_save + * + * @param Shipment $shipment The shipment object being saved. + * @param boolean $is_new Indicator to determine whether this is a new shipment or not. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( "woocommerce_gzd_{$hook_postfix}shipment_after_save", $this, $is_new ); + + $this->status_transition(); + $this->reset_content_data(); + + } catch ( Exception $e ) { + /** + * This is a tweak to prevent the WooCommerce PayPal Payments Plugin compatibility script + * from breaking our code in case an error occurs while transmitting tracking data to PayPal. + * This tweak should only be included as long as the bug persists. + * @TODO Check whether the issue persists in next release cycles + * + * @see https://github.com/woocommerce/woocommerce-paypal-payments/issues/1020 + */ + if ( is_a( $e, 'WooCommerce\PayPalCommerce\ApiClient\Exception\PayPalApiException' ) || is_a( $e, 'WooCommerce\PayPalCommerce\ApiClient\Exception\RuntimeException' ) ) { + $this->status_transition(); + $this->reset_content_data(); + } else { + $logger = wc_get_logger(); + $logger->error( + sprintf( 'Error saving shipment #%d', $this->get_id() ), + array( + 'shipment' => $this, + 'error' => $e, + ) + ); + } + } + + return $this->get_id(); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ShipmentError.php b/packages/woocommerce-germanized-shipments/src/ShipmentError.php new file mode 100644 index 000000000..da87055f4 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShipmentError.php @@ -0,0 +1,88 @@ +set_is_soft_error(); + } + + public function add( $code, $message, $data = '' ) { + parent::add( $code, $message, $data ); + + $this->set_is_soft_error(); + } + + public function get_error_messages_by_type() { + $errors = $this->get_error_messages(); + $soft = $this->get_soft_error_messages(); + + return array( + 'error' => array_diff( $errors, $soft ), + 'soft' => $soft, + ); + } + + public function get_soft_error_messages( $code = '' ) { + // Return all messages if no code specified. + if ( empty( $code ) ) { + $all_messages = array(); + foreach ( (array) $this->errors as $code => $messages ) { + $data = $this->get_error_data( $code ); + + if ( is_string( $data ) && 'soft' === $data ) { + $all_messages = array_merge( $all_messages, $messages ); + } + } + + return $all_messages; + } + + if ( isset( $this->errors[ $code ] ) ) { + $data = $this->get_error_data( $code ); + + if ( is_string( $data ) && 'soft' === $data ) { + return $this->errors[ $code ]; + } + + return array(); + } else { + return array(); + } + } + + public function add_soft_error( $code, $message ) { + parent::add( $code, $message, 'soft' ); + } + + public function is_soft_error() { + return $this->is_soft_error; + } + + protected function set_is_soft_error() { + $is_soft_error = true; + + foreach ( $this->get_error_codes() as $code ) { + $error_data = $this->get_error_data( $code ); + + if ( ! is_string( $error_data ) || 'soft' !== $error_data ) { + $is_soft_error = false; + break; + } + } + + $this->is_soft_error = $is_soft_error; + } + + public static function from_wp_error( \WP_Error $from ) { + $error = new self(); + $error->merge_from( $from ); + + return $error; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ShipmentFactory.php b/packages/woocommerce-germanized-shipments/src/ShipmentFactory.php new file mode 100644 index 000000000..f6595c8e9 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShipmentFactory.php @@ -0,0 +1,88 @@ +get_shipment_type( $shipment_id ); + + /** + * Shipment type cannot be found, seems to not exist. + */ + if ( empty( $shipment_type ) ) { + return false; + } + + $shipment_type_data = wc_gzd_get_shipment_type_data( $shipment_type ); + } else { + $shipment_type_data = wc_gzd_get_shipment_type_data( $shipment_type ); + } + + if ( $shipment_type_data ) { + $classname = $shipment_type_data['class_name']; + } else { + $classname = false; + } + + /** + * Filter to adjust the classname used to construct a Shipment. + * + * @param string $clasname The classname to be used. + * @param integer $shipment_id The shipment id. + * @param string $shipment_type The shipment type. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $classname = apply_filters( 'woocommerce_gzd_shipment_class', $classname, $shipment_id, $shipment_type ); + + if ( ! class_exists( $classname ) ) { + return false; + } + + try { + return new $classname( $shipment_id ); + } catch ( Exception $e ) { + wc_caught_exception( $e, __FUNCTION__, array( $shipment_id, $shipment_type ) ); + return false; + } + } + + public static function get_shipment_id( $shipment ) { + if ( is_numeric( $shipment ) ) { + return $shipment; + } elseif ( $shipment instanceof Shipment ) { + return $shipment->get_id(); + } elseif ( ! empty( $shipment->shipment_id ) ) { + return $shipment->shipment_id; + } else { + return false; + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ShipmentItem.php b/packages/woocommerce-germanized-shipments/src/ShipmentItem.php new file mode 100644 index 000000000..4b3fe3de3 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShipmentItem.php @@ -0,0 +1,605 @@ + 0, + 'order_item_id' => 0, + 'parent_id' => 0, + 'quantity' => 1, + 'product_id' => 0, + 'weight' => '', + 'width' => '', + 'length' => '', + 'height' => '', + 'sku' => '', + 'name' => '', + 'total' => 0, + 'subtotal' => 0, + 'hs_code' => '', + 'manufacture_country' => '', + 'attributes' => array(), + ); + + /** + * Stores meta in cache for future reads. + * A group must be set to to enable caching. + * + * @var string + */ + protected $cache_group = 'shipment-items'; + + /** + * Meta type. This should match up with + * the types available at https://developer.wordpress.org/reference/functions/add_metadata/. + * WP defines 'post', 'user', 'comment', and 'term'. + * + * @var string + */ + protected $meta_type = 'shipment_item'; + + /** + * This is the name of this object type. + * + * @var string + */ + protected $object_type = 'shipment_item'; + + /** + * Constructor. + * + * @param int|object|array $item ID to load from the DB, or WC_Order_Item object. + */ + public function __construct( $item = 0 ) { + parent::__construct( $item ); + + if ( $item instanceof ShipmentItem ) { + $this->set_id( $item->get_id() ); + } elseif ( is_numeric( $item ) && $item > 0 ) { + $this->set_id( $item ); + } else { + $this->set_object_read( true ); + } + + $this->data_store = WC_Data_Store::load( 'shipment-item' ); + + // If we have an ID, load the user from the DB. + if ( $this->get_id() ) { + try { + $this->data_store->read( $this ); + } catch ( Exception $e ) { + $this->set_id( 0 ); + $this->set_object_read( true ); + } + } else { + $this->set_object_read( true ); + } + } + + /** + * Merge changes with data and clear. + * Overrides WC_Data::apply_changes. + * array_replace_recursive does not work well for order items because it merges taxes instead + * of replacing them. + * + * @since 3.2.0 + */ + public function apply_changes() { + if ( function_exists( 'array_replace' ) ) { + $this->data = array_replace( $this->data, $this->changes ); // phpcs:ignore PHPCompatibility.FunctionUse.NewFunctions.array_replaceFound + } else { // PHP 5.2 compatibility. + foreach ( $this->changes as $key => $change ) { + $this->data[ $key ] = $change; + } + } + $this->changes = array(); + } + + /* + |-------------------------------------------------------------------------- + | Getters + |-------------------------------------------------------------------------- + */ + + public function get_type() { + return 'simple'; + } + + /** + * Get order ID this meta belongs to. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return int + */ + public function get_shipment_id( $context = 'view' ) { + return $this->get_prop( 'shipment_id', $context ); + } + + /** + * Get order ID this meta belongs to. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return int + */ + public function get_order_item_id( $context = 'view' ) { + return $this->get_prop( 'order_item_id', $context ); + } + + /** + * Get order ID this meta belongs to. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return int + */ + public function get_product_id( $context = 'view' ) { + return $this->get_prop( 'product_id', $context ); + } + + /** + * Get item parent id. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return int + */ + public function get_parent_id( $context = 'view' ) { + return $this->get_prop( 'parent_id', $context ); + } + + /** + * Get order ID this meta belongs to. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return int + */ + public function get_total( $context = 'view' ) { + return $this->get_prop( 'total', $context ); + } + + /** + * Get order ID this meta belongs to. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return int + */ + public function get_subtotal( $context = 'view' ) { + $subtotal = $this->get_prop( 'subtotal', $context ); + + if ( 'view' === $context && empty( $subtotal ) ) { + $subtotal = $this->get_total(); + } + + return $subtotal; + } + + /** + * Get quantity. + * + * @return int + */ + public function get_sku( $context = 'view' ) { + return $this->get_prop( 'sku', $context ); + } + + /** + * Get quantity. + * + * @return int + */ + public function get_quantity( $context = 'view' ) { + return $this->get_prop( 'quantity', $context ); + } + + /** + * Get weight. + * + * @return string + */ + public function get_weight( $context = 'view' ) { + return $this->get_prop( 'weight', $context ); + } + + /** + * Get width. + * + * @return string + */ + public function get_width( $context = 'view' ) { + return $this->get_prop( 'width', $context ); + } + + /** + * Get length. + * + * @return string + */ + public function get_length( $context = 'view' ) { + return $this->get_prop( 'length', $context ); + } + + /** + * Get height. + * + * @return string + */ + public function get_height( $context = 'view' ) { + return $this->get_prop( 'height', $context ); + } + + public function get_name( $context = 'view' ) { + $name = $this->get_prop( 'name', $context ); + + if ( 'view' === $context && empty( $name ) && ( $item = $this->get_order_item() ) ) { + $name = $item->get_name(); + } + + return $name; + } + + public function get_hs_code( $context = 'view' ) { + $legacy = $this->get_meta( '_dhl_hs_code', $context ); + $prop = $this->get_prop( 'hs_code', $context ); + + if ( '' === $prop && ! empty( $legacy ) ) { + $prop = $legacy; + } + + return $prop; + } + + public function get_manufacture_country( $context = 'view' ) { + $legacy = $this->get_meta( '_dhl_manufacture_country', $context ); + $prop = $this->get_prop( 'manufacture_country', $context ); + + if ( '' === $prop && ! empty( $legacy ) ) { + $prop = $legacy; + } + + return $prop; + } + + /** + * Get attributes. + * + * @return string[] + */ + public function get_attributes( $context = 'view' ) { + return $this->get_prop( 'attributes', $context ); + } + + public function has_attributes() { + $attributes = $this->get_attributes(); + + return ! empty( $attributes ); + } + + /** + * Get parent order object. + * + * @return SimpleShipment|ReturnShipment|Shipment + */ + public function get_shipment() { + if ( is_null( $this->shipment ) && 0 < $this->get_shipment_id() ) { + $this->shipment = wc_gzd_get_shipment( $this->get_shipment_id() ); + } + + $shipment = ( $this->shipment ) ? $this->shipment : false; + + return $shipment; + } + + /** + * Sets the linked shipment instance. + * + * @param Shipment $shipment + */ + public function set_shipment( &$shipment ) { + $this->shipment = $shipment; + } + + /** + * Syncs an item with either it's parent item or the corresponding order item. + * + * @param array $args + */ + public function sync( $args = array() ) { + $item = false; + + if ( $shipment = $this->get_shipment() ) { + if ( 'return' === $shipment->get_type() ) { + if ( $shipment = $this->get_shipment() ) { + if ( $order_shipment = $shipment->get_order_shipment() ) { + $item = $order_shipment->get_simple_shipment_item( $this->get_order_item_id() ); + } + } + } else { + $item = $this->get_order_item(); + } + } + + if ( is_a( $item, '\Vendidero\Germanized\Shipments\ShipmentItem' ) ) { + + $default_data = $item->get_data(); + + unset( $default_data['id'] ); + unset( $default_data['shipment_id'] ); + + $default_data['parent_id'] = $item->get_id(); + $args = wp_parse_args( $args, $default_data ); + + } elseif ( is_a( $item, 'WC_Order_Item' ) ) { + + if ( is_callable( array( $item, 'get_variation_id' ) ) && is_callable( array( $item, 'get_product_id' ) ) ) { + $this->set_product_id( $item->get_variation_id() ? $item->get_variation_id() : $item->get_product_id() ); + } elseif ( is_callable( array( $item, 'get_product_id' ) ) ) { + $this->set_product_id( $item->get_product_id() ); + } + + $args = wp_parse_args( + $args, + array( + 'quantity' => 1, + ) + ); + + $product = $this->get_product(); + $s_product = wc_gzd_shipments_get_product( $product ); + + /** + * Calculate the order item total per unit to make sure it is independent from + * shipment item quantity. + */ + $tax_total = is_callable( array( $item, 'get_total_tax' ) ) ? ( (float) $item->get_total_tax() / $item->get_quantity() ) * $args['quantity'] : 0; + $total = is_callable( array( $item, 'get_total' ) ) ? ( (float) $item->get_total() / $item->get_quantity() ) * $args['quantity'] : 0; + $subtotal = is_callable( array( $item, 'get_subtotal' ) ) ? (float) $item->get_subtotal() / $item->get_quantity() * $args['quantity'] : 0; + $tax_subtotal = is_callable( array( $item, 'get_subtotal_tax' ) ) ? (float) $item->get_subtotal_tax() / $item->get_quantity() * $args['quantity'] : 0; + $meta = $item->get_formatted_meta_data( apply_filters( "{$this->get_hook_prefix()}hide_meta_prefix", '_', $this ), apply_filters( "{$this->get_hook_prefix()}include_all_meta", false, $this ) ); + $attributes = array(); + + foreach ( $meta as $meta_id => $entry ) { + $attributes[] = array( + 'key' => $entry->key, + 'value' => str_replace( array( '

', '

' ), '', $entry->display_value ), + 'label' => $entry->display_key, + 'order_item_meta_id' => $meta_id, + ); + } + + $args = wp_parse_args( + $args, + array( + 'order_item_id' => $item->get_id(), + 'quantity' => 1, + 'name' => $item->get_name(), + 'sku' => $product ? $product->get_sku() : '', + 'total' => $total + $tax_total, + 'subtotal' => $subtotal + $tax_subtotal, + 'weight' => $product ? wc_get_weight( $product->get_weight(), $shipment->get_weight_unit() ) : '', + 'length' => $product ? wc_get_dimension( $product->get_length(), $shipment->get_dimension_unit() ) : '', + 'width' => $product ? wc_get_dimension( $product->get_width(), $shipment->get_dimension_unit() ) : '', + 'height' => $product ? wc_get_dimension( $product->get_height(), $shipment->get_dimension_unit() ) : '', + 'hs_code' => $s_product ? $s_product->get_hs_code() : '', + 'manufacture_country' => $s_product ? $s_product->get_manufacture_country() : '', + 'attributes' => $attributes, + ) + ); + } + + $this->set_props( $args ); + + /** + * Action that fires after a shipment item has been synced. Syncing is used to + * keep the shipment item in sync with the corresponding order item or parent shipment item. + * + * @param WC_Order_Item|ShipmentItem $item The order item object or parent shipment item. + * @param array $args Array containing props in key => value pairs which have been updated. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_item_synced', $this, $item, $args ); + } + + public function get_order_item() { + if ( is_null( $this->order_item ) && 0 < $this->get_order_item_id() ) { + if ( $shipment = $this->get_shipment() ) { + + if ( $order = $shipment->get_order() ) { + $this->order_item = $order->get_item( $this->get_order_item_id() ); + } + } + } + + $item = ( $this->order_item ) ? $this->order_item : false; + + return $item; + } + + public function get_product() { + if ( is_null( $this->product ) && 0 < $this->get_product_id() ) { + $this->product = wc_get_product( $this->get_product_id() ); + } + + $product = ( $this->product ) ? $this->product : false; + + return $product; + } + + /* + |-------------------------------------------------------------------------- + | Setters + |-------------------------------------------------------------------------- + */ + + /** + * Set order ID. + * + * @param int $value Order ID. + */ + public function set_shipment_id( $value ) { + $this->order_item = null; + $this->shipment = null; + + $this->set_prop( 'shipment_id', absint( $value ) ); + } + + /** + * Set order ID. + * + * @param int $value Order ID. + */ + public function set_order_item_id( $value ) { + $this->set_prop( 'order_item_id', absint( $value ) ); + } + + /** + * Set order ID. + * + * @param int $value Order ID. + */ + public function set_total( $value ) { + $value = wc_format_decimal( $value ); + + if ( ! is_numeric( $value ) ) { + $value = 0; + } + + $this->set_prop( 'total', $value ); + } + + /** + * Set order ID. + * + * @param int $value Order ID. + */ + public function set_subtotal( $value ) { + $value = wc_format_decimal( $value ); + + if ( ! is_numeric( $value ) ) { + $value = 0; + } + + $this->set_prop( 'subtotal', $value ); + } + + /** + * Set order ID. + * + * @param int $value Order ID. + */ + public function set_product_id( $value ) { + $this->product = null; + $this->set_prop( 'product_id', absint( $value ) ); + } + + /** + * Set parent id. + * + * @param int $value parent id. + */ + public function set_parent_id( $value ) { + $this->set_prop( 'parent_id', absint( $value ) ); + } + + public function set_sku( $sku ) { + $this->set_prop( 'sku', $sku ); + } + + /** + * Set weight in kg + * + * @param $weight + */ + public function set_weight( $weight ) { + $this->set_prop( 'weight', '' === $weight ? '' : wc_format_decimal( $weight ) ); + } + + /** + * Set width in cm + * + * @param $weight + */ + public function set_width( $width ) { + $this->set_prop( 'width', '' === $width ? '' : wc_format_decimal( $width ) ); + } + + /** + * Set length in cm + * + * @param $weight + */ + public function set_length( $length ) { + $this->set_prop( 'length', '' === $length ? '' : wc_format_decimal( $length ) ); + } + + /** + * Set height in cm + * + * @param $weight + */ + public function set_height( $height ) { + $this->set_prop( 'height', '' === $height ? '' : wc_format_decimal( $height ) ); + } + + public function get_dimensions( $context = 'view' ) { + return array( + 'length' => $this->get_length( $context ), + 'width' => $this->get_width( $context ), + 'height' => $this->get_height( $context ), + ); + } + + public function set_quantity( $quantity ) { + $this->set_prop( 'quantity', absint( $quantity ) ); + } + + public function set_name( $name ) { + $this->set_prop( 'name', $name ); + } + + public function set_hs_code( $code ) { + $this->set_prop( 'hs_code', $code ); + } + + public function set_manufacture_country( $country ) { + $this->set_prop( 'manufacture_country', wc_strtoupper( $country ) ); + } + + /** + * Set attributes + * + * @param $attributes + */ + public function set_attributes( $attributes ) { + $this->set_prop( 'attributes', (array) $attributes ); + } + + /* + |-------------------------------------------------------------------------- + | Other Methods + |-------------------------------------------------------------------------- + */ +} diff --git a/packages/woocommerce-germanized-shipments/src/ShipmentQuery.php b/packages/woocommerce-germanized-shipments/src/ShipmentQuery.php new file mode 100644 index 000000000..54c045a17 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShipmentQuery.php @@ -0,0 +1,563 @@ + array_keys( wc_gzd_get_shipment_statuses() ), + 'limit' => 10, + 'order_id' => '', + 'parent_id' => '', + 'product_ids' => '', + 'type' => 'simple', + 'country' => '', + 'tracking_id' => '', + 'order' => 'DESC', + 'orderby' => 'date_created', + 'shipping_provider' => '', + 'return' => 'objects', + 'page' => 1, + 'offset' => '', + 'paginate' => false, + 'search' => '', + 'search_columns' => array(), + ); + } + + /** + * Get shipments matching the current query vars. + * + * @return Shipment[] Array containing Shipments. + */ + public function get_shipments() { + /** + * Filter to adjust query arguments passed to a Shipment query. + * + * @param array $args The arguments passed. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $args = apply_filters( 'woocommerce_gzd_shipment_query_args', $this->get_query_vars() ); + $args = WC_Data_Store::load( 'shipment' )->get_query_args( $args ); + + $this->query( $args ); + + /** + * Filter to adjust the Shipment query result. + * + * @param Shipment[] $results Shipment results. + * @param array $args The arguments passed. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipment_query', $this->results, $args ); + } + + public function get_total() { + return $this->total_shipments; + } + + public function get_max_num_pages() { + return $this->max_num_pages; + } + + /** + * Query shipments. + * + * @param array $query_args + */ + protected function query( $query_args ) { + global $wpdb; + + $this->args = $query_args; + $this->parse_query(); + $this->prepare_query(); + + $qv =& $this->args; + + $this->results = null; + + if ( null === $this->results ) { + $this->request = "SELECT $this->query_fields $this->query_from $this->query_where $this->query_orderby $this->query_limit"; + + if ( is_array( $qv['fields'] ) || 'objects' === $qv['fields'] ) { + $this->results = $wpdb->get_results( $this->request ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } else { + $this->results = $wpdb->get_col( $this->request ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + } + + if ( isset( $qv['count_total'] ) && $qv['count_total'] ) { + $found_shipments_query = 'SELECT FOUND_ROWS()'; + $this->total_shipments = (int) $wpdb->get_var( $found_shipments_query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared + $this->max_num_pages = ceil( $this->total_shipments / $qv['posts_per_page'] ); + } + } + + if ( ! $this->results ) { + return; + } + + if ( 'objects' === $qv['fields'] ) { + foreach ( $this->results as $key => $shipment ) { + $this->results[ $key ] = wc_gzd_get_shipment( $shipment ); + } + } + } + + /** + * Parse the query before preparing it. + */ + protected function parse_query() { + if ( isset( $this->args['order_id'] ) ) { + $this->args['order_id'] = absint( $this->args['order_id'] ); + } + + if ( isset( $this->args['shipping_provider'] ) ) { + $this->args['shipping_provider'] = wc_clean( $this->args['shipping_provider'] ); + } + + if ( isset( $this->args['parent_id'] ) ) { + $this->args['parent_id'] = absint( $this->args['parent_id'] ); + } + + if ( isset( $this->args['product_ids'] ) ) { + $this->args['product_ids'] = (array) $this->args['product_ids']; + $this->args['product_ids'] = array_map( 'absint', $this->args['product_ids'] ); + } + + if ( isset( $this->args['tracking_id'] ) ) { + $this->args['tracking_id'] = sanitize_key( $this->args['tracking_id'] ); + } + + if ( isset( $this->args['status'] ) ) { + $this->args['status'] = (array) $this->args['status']; + $this->args['status'] = array_map( 'sanitize_key', $this->args['status'] ); + } + + if ( isset( $this->args['type'] ) ) { + $this->args['type'] = (array) $this->args['type']; + $this->args['type'] = array_map( 'wc_clean', $this->args['type'] ); + } + + if ( isset( $this->args['country'] ) ) { + $countries = isset( WC()->countries ) ? WC()->countries : false; + + if ( $countries && is_a( $countries, 'WC_Countries' ) ) { + + // Reverse search by country name + if ( $key = array_search( $this->args['country'], $countries->get_countries(), true ) ) { + $this->args['country'] = $key; + } + } + + // Country Code ISO + $this->args['country'] = strtoupper( substr( $this->args['country'], 0, 2 ) ); + } + + if ( isset( $this->args['search'] ) ) { + $this->args['search'] = wc_clean( $this->args['search'] ); + + if ( ! isset( $this->args['search_columns'] ) ) { + $this->args['search_columns'] = array(); + } + } + + if ( isset( $this->args['orderby'] ) ) { + if ( 'weight' === $this->args['orderby'] ) { + $this->args['meta_query'][] = array( + 'relation' => 'OR', + array( + 'key' => '_weight', + 'compare' => 'NOT EXISTS', + ), + array( + 'key' => '_weight', + 'compare' => '>=', + 'value' => 0, + ), + ); + + $this->args['orderby'] = 'meta_value_num'; + } + } + } + + /** + * Prepare the query for DB usage. + */ + protected function prepare_query() { + global $wpdb; + + if ( is_array( $this->args['fields'] ) ) { + $this->args['fields'] = array_unique( $this->args['fields'] ); + + $this->query_fields = array(); + + foreach ( $this->args['fields'] as $field ) { + $field = 'ID' === $field ? 'shipment_id' : sanitize_key( $field ); + $this->query_fields[] = "$wpdb->gzd_shipments.$field"; + } + + $this->query_fields = implode( ',', $this->query_fields ); + + } elseif ( 'objects' === $this->args['fields'] ) { + $this->query_fields = "$wpdb->gzd_shipments.*"; + } else { + $this->query_fields = "$wpdb->gzd_shipments.shipment_id"; + } + + if ( isset( $this->args['count_total'] ) && $this->args['count_total'] ) { + $this->query_fields = 'SQL_CALC_FOUND_ROWS ' . $this->query_fields; + } + + $this->query_from = "FROM $wpdb->gzd_shipments"; + $this->query_where = 'WHERE 1=1'; + + // order id + if ( isset( $this->args['order_id'] ) ) { + $this->query_where .= $wpdb->prepare( ' AND shipment_order_id = %d', $this->args['order_id'] ); + } + + // order id + if ( isset( $this->args['shipping_provider'] ) ) { + $this->query_where .= $wpdb->prepare( ' AND shipment_shipping_provider = %s', $this->args['shipping_provider'] ); + } + + // tracking id + if ( isset( $this->args['tracking_id'] ) ) { + $this->query_where .= $wpdb->prepare( " AND shipment_tracking_id IN ('%s')", $this->args['tracking_id'] ); // phpcs:ignore WordPress.DB.PreparedSQLPlaceholders.QuotedSimplePlaceholder + } + + // parent id + if ( isset( $this->args['parent_id'] ) ) { + $this->query_where .= $wpdb->prepare( ' AND shipment_parent_id = %d', $this->args['parent_id'] ); + } + + // product ids + if ( isset( $this->args['product_ids'] ) ) { + $product_ids_placeholders = implode( ', ', array_fill( 0, count( $this->args['product_ids'] ), '%d' ) ); + + $this->query_from .= " JOIN {$wpdb->prefix}woocommerce_gzd_shipment_items as shipment_items ON ( shipment_items.shipment_id = {$wpdb->prefix}woocommerce_gzd_shipments.shipment_id ) "; + $this->query_where .= $wpdb->prepare( " AND shipment_items.shipment_item_product_id IN ({$product_ids_placeholders})", $this->args['product_ids'] ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.PreparedSQLPlaceholders.UnfinishedPrepare + } + + // country + if ( isset( $this->args['country'] ) ) { + $this->query_where .= $wpdb->prepare( " AND shipment_country IN ('%s')", $this->args['country'] ); // phpcs:ignore WordPress.DB.PreparedSQLPlaceholders.QuotedSimplePlaceholder + } + + // type + if ( isset( $this->args['type'] ) ) { + $types = $this->args['type']; + $p_types = array(); + + foreach ( $types as $type ) { + $p_types[] = $wpdb->prepare( 'shipment_type = %s', $type ); + } + + $where_type = implode( ' OR ', $p_types ); + + if ( ! empty( $where_type ) ) { + $this->query_where .= " AND ($where_type)"; + } + } + + // status + if ( isset( $this->args['status'] ) ) { + $stati = $this->args['status']; + $p_status = array(); + + foreach ( $stati as $status ) { + $p_status[] = $wpdb->prepare( 'shipment_status = %s', $status ); + } + + $where_status = implode( ' OR ', $p_status ); + + if ( ! empty( $where_status ) ) { + $this->query_where .= " AND ($where_status)"; + } + } + + // Search + $search = ''; + + if ( isset( $this->args['search'] ) ) { + $search = trim( $this->args['search'] ); + } + + if ( $search ) { + + $leading_wild = ( ltrim( $search, '*' ) !== $search ); + $trailing_wild = ( rtrim( $search, '*' ) !== $search ); + + if ( $leading_wild && $trailing_wild ) { + $wild = 'both'; + } elseif ( $leading_wild ) { + $wild = 'leading'; + } elseif ( $trailing_wild ) { + $wild = 'trailing'; + } else { + $wild = false; + } + if ( $wild ) { + $search = trim( $search, '*' ); + } + + $search_columns = array(); + + if ( $this->args['search_columns'] ) { + $search_columns = array_intersect( $this->args['search_columns'], array( 'shipment_id', 'shipment_country', 'shipment_tracking_id', 'shipment_order_id', 'shipment_shipping_provider', 'shipment_shipping_method', 'shipment_search_index' ) ); + } + + if ( ! $search_columns ) { + if ( is_numeric( $search ) ) { + $search_columns = array( 'shipment_id', 'shipment_order_id', 'shipment_tracking_id' ); + } elseif ( strlen( $search ) === 2 ) { + $search_columns = array( 'shipment_country' ); + } else { + $search_columns = array( 'shipment_id', 'shipment_country', 'shipment_tracking_id', 'shipment_order_id', 'shipment_search_index' ); + } + } + + /** + * Filters the columns to search in a ShipmentQuery search. + * + * The default columns depend on the search term, and include 'shipment_id', 'shipment_country', + * 'shipment_tracking_id', 'shipment_order_id', 'shipment_shipping_provider' and 'shipment_shipping_method'. + * + * @since 3.0.0 + * + * @param string[] $search_columns Array of column names to be searched. + * @param string $search Text being searched. + * @param ShipmentQuery $this The current ShipmentQuery instance. + * + * @package Vendidero/Germanized/Shipments + */ + $search_columns = apply_filters( 'woocommerce_gzd_shipment_search_columns', $search_columns, $search, $this ); + + $this->query_where .= $this->get_search_sql( $search, $search_columns, $wild ); + } + + // Parse and sanitize 'include', for use by 'orderby' as well as 'include' below. + if ( ! empty( $this->args['include'] ) ) { + $include = wp_parse_id_list( $this->args['include'] ); + } else { + $include = false; + } + + // Meta query. + $this->meta_query = new WP_Meta_Query(); + $this->meta_query->parse_query_vars( $this->args ); + + if ( ! empty( $this->meta_query->queries ) ) { + $clauses = $this->meta_query->get_sql( 'gzd_shipment', $wpdb->gzd_shipments, 'shipment_id', $this ); + $this->query_from .= $clauses['join']; + $this->query_where .= $clauses['where']; + + if ( $this->meta_query->has_or_relation() ) { + $this->query_fields = 'DISTINCT ' . $this->query_fields; + } + } + + // sorting + $this->args['order'] = isset( $this->args['order'] ) ? strtoupper( $this->args['order'] ) : ''; + $order = $this->parse_order( $this->args['order'] ); + + if ( empty( $this->args['orderby'] ) ) { + // Default order is by 'user_login'. + $ordersby = array( 'date_created' => $order ); + } elseif ( is_array( $this->args['orderby'] ) ) { + $ordersby = $this->args['orderby']; + } else { + // 'orderby' values may be a comma- or space-separated list. + $ordersby = preg_split( '/[,\s]+/', $this->args['orderby'] ); + } + + $orderby_array = array(); + + foreach ( $ordersby as $_key => $_value ) { + if ( ! $_value ) { + continue; + } + + if ( is_int( $_key ) ) { + // Integer key means this is a flat array of 'orderby' fields. + $_orderby = $_value; + $_order = $order; + } else { + // Non-integer key means this the key is the field and the value is ASC/DESC. + $_orderby = $_key; + $_order = $_value; + } + + $parsed = $this->parse_orderby( $_orderby ); + + if ( ! $parsed ) { + continue; + } + + $orderby_array[] = $parsed . ' ' . $this->parse_order( $_order ); + } + + // If no valid clauses were found, order by user_login. + if ( empty( $orderby_array ) ) { + $orderby_array[] = "shipment_id $order"; + } + + $this->query_orderby = 'ORDER BY ' . implode( ', ', $orderby_array ); + + // limit + if ( isset( $this->args['posts_per_page'] ) && $this->args['posts_per_page'] > 0 ) { + if ( isset( $this->args['offset'] ) ) { + $this->query_limit = $wpdb->prepare( 'LIMIT %d, %d', $this->args['offset'], $this->args['posts_per_page'] ); + } else { + $this->query_limit = $wpdb->prepare( 'LIMIT %d, %d', $this->args['posts_per_page'] * ( $this->args['page'] - 1 ), $this->args['posts_per_page'] ); + } + } + + if ( ! empty( $include ) ) { + // Sanitized earlier. + $ids = implode( ',', $include ); + $this->query_where .= " AND $wpdb->gzd_shipments.shipment_id IN ($ids)"; + } elseif ( ! empty( $this->args['exclude'] ) ) { + $ids = implode( ',', wp_parse_id_list( $this->args['exclude'] ) ); + $this->query_where .= " AND $wpdb->gzd_shipments.shipment_id NOT IN ($ids)"; + } + + // Date queries are allowed for the user_registered field. + if ( ! empty( $this->args['date_query'] ) && is_array( $this->args['date_query'] ) ) { + $date_query = new WP_Date_Query( $this->args['date_query'], 'shipment_date_created' ); + $this->query_where .= $date_query->get_sql(); + } + } + + /** + * Used internally to generate an SQL string for searching across multiple columns + * + * @since 3.0.6 + * + * @global wpdb $wpdb WordPress database abstraction object. + * + * @param string $string + * @param array $cols + * @param bool $wild Whether to allow wildcard searches. Default is false for Network Admin, true for single site. + * Single site allows leading and trailing wildcards, Network Admin only trailing. + * @return string + */ + protected function get_search_sql( $string, $cols, $wild = false ) { + global $wpdb; + + $searches = array(); + $leading_wild = ( 'leading' === $wild || 'both' === $wild ) ? '%' : ''; + $trailing_wild = ( 'trailing' === $wild || 'both' === $wild ) ? '%' : ''; + $like = $leading_wild . $wpdb->esc_like( $string ) . $trailing_wild; + + foreach ( $cols as $col ) { + if ( 'ID' === $col ) { + $searches[] = $wpdb->prepare( "$wpdb->gzd_shipments.$col = %s", $string ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared + } else { + $searches[] = $wpdb->prepare( "$wpdb->gzd_shipments.$col LIKE %s", $like ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared + } + } + + return ' AND (' . implode( ' OR ', $searches ) . ')'; + } + + /** + * Parse orderby statement. + * + * @param string $orderby + * @return string + */ + protected function parse_orderby( $orderby ) { + global $wpdb; + + $meta_query_clauses = $this->meta_query->get_clauses(); + $_orderby = ''; + + if ( in_array( $orderby, array( 'country', 'status', 'tracking_id', 'date_created', 'order_id' ), true ) ) { + $_orderby = 'shipment_' . $orderby; + } elseif ( 'date' === $orderby ) { + $_orderby = 'shipment_date_created'; + } elseif ( 'ID' === $orderby || 'id' === $orderby ) { + $_orderby = 'shipment_id'; + } elseif ( 'meta_value' === $orderby || $this->get( 'meta_key' ) === $orderby ) { + $_orderby = "$wpdb->gzd_shipmentmeta.meta_value"; + } elseif ( 'meta_value_num' === $orderby ) { + $_orderby = "$wpdb->gzd_shipmentmeta.meta_value+0"; + } elseif ( 'include' === $orderby && ! empty( $this->args['include'] ) ) { + $include = wp_parse_id_list( $this->args['include'] ); + $include_sql = implode( ',', $include ); + $_orderby = "FIELD( $wpdb->gzd_shipments.shipment_id, $include_sql )"; + } elseif ( isset( $meta_query_clauses[ $orderby ] ) ) { + $meta_clause = $meta_query_clauses[ $orderby ]; + $_orderby = sprintf( 'CAST(%s.meta_value AS %s)', esc_sql( $meta_clause['alias'] ), esc_sql( $meta_clause['cast'] ) ); + } + + return $_orderby; + } + + /** + * Parse order statement. + * + * @param string $order + * @return string + */ + protected function parse_order( $order ) { + if ( ! is_string( $order ) || empty( $order ) ) { + return 'DESC'; + } + + if ( 'ASC' === strtoupper( $order ) ) { + return 'ASC'; + } else { + return 'DESC'; + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ShipmentReturnItem.php b/packages/woocommerce-germanized-shipments/src/ShipmentReturnItem.php new file mode 100644 index 000000000..14bba66d1 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShipmentReturnItem.php @@ -0,0 +1,32 @@ + '', + ); + + public function get_type() { + return 'return'; + } + + public function get_return_reason_code( $context = 'view' ) { + return $this->get_prop( 'return_reason_code', $context ); + } + + public function set_return_reason_code( $code ) { + $this->set_prop( 'return_reason_code', $code ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ShippingProvider/Auto.php b/packages/woocommerce-germanized-shipments/src/ShippingProvider/Auto.php new file mode 100644 index 000000000..6439fe820 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShippingProvider/Auto.php @@ -0,0 +1,570 @@ + '', + 'label_minimum_shipment_weight' => '', + 'label_auto_enable' => false, + 'label_auto_shipment_status' => 'gzd-processing', + 'label_return_auto_enable' => false, + 'label_return_auto_shipment_status' => 'gzd-processing', + 'label_auto_shipment_status_shipped' => false, + ); + + public function get_label_default_shipment_weight( $context = 'view' ) { + $weight = $this->get_prop( 'label_default_shipment_weight', $context ); + + if ( 'view' === $context && '' === $weight ) { + $weight = $this->get_default_label_default_shipment_weight(); + } + + return $weight; + } + + protected function get_default_label_default_shipment_weight() { + return 0; + } + + public function get_label_minimum_shipment_weight( $context = 'view' ) { + $weight = $this->get_prop( 'label_minimum_shipment_weight', $context ); + + if ( 'view' === $context && '' === $weight ) { + $weight = $this->get_default_label_minimum_shipment_weight(); + } + + return $weight; + } + + protected function get_default_label_minimum_shipment_weight() { + return 0.5; + } + + /** + * @param false|Shipment $shipment + * + * @return boolean + */ + public function automatically_generate_label( $shipment = false ) { + $setting_key = 'label_auto_enable'; + + if ( $shipment ) { + if ( 'return' === $shipment->get_type() ) { + $setting_key = 'label_return_auto_enable'; + } + + return wc_string_to_bool( $this->get_shipment_setting( $shipment, $setting_key, false ) ); + } else { + return wc_string_to_bool( $this->get_setting( $setting_key, false ) ); + } + } + + /** + * @param false|Shipment $shipment + * + * @return string + */ + public function get_label_automation_shipment_status( $shipment = false ) { + $setting_key = 'label_auto_shipment_status'; + + if ( $shipment ) { + if ( 'return' === $shipment->get_type() ) { + $setting_key = 'label_return_auto_shipment_status'; + } + + return $this->get_shipment_setting( $shipment, $setting_key, 'gzd-processing' ); + } else { + return $this->get_setting( $setting_key, 'gzd-processing' ); + } + } + + public function automatically_set_shipment_status_shipped( $shipment = false ) { + $setting_key = 'label_auto_shipment_status_shipped'; + + if ( $shipment ) { + return wc_string_to_bool( $this->get_shipment_setting( $shipment, $setting_key, false ) ); + } else { + return wc_string_to_bool( $this->get_setting( $setting_key, false ) ); + } + } + + public function get_label_auto_enable( $context = 'view' ) { + return $this->get_prop( 'label_auto_enable', $context ); + } + + public function get_label_auto_shipment_status_shipped( $context = 'view' ) { + return $this->get_prop( 'label_auto_shipment_status_shipped', $context ); + } + + public function get_label_auto_shipment_status( $context = 'view' ) { + return $this->get_prop( 'label_auto_shipment_status', $context ); + } + + public function automatically_generate_return_label() { + return $this->get_label_return_auto_enable(); + } + + public function get_label_return_auto_enable( $context = 'view' ) { + return $this->get_prop( 'label_return_auto_enable', $context ); + } + + public function get_label_return_auto_shipment_status( $context = 'view' ) { + return $this->get_prop( 'label_return_auto_shipment_status', $context ); + } + + public function is_sandbox() { + return false; + } + + public function set_label_default_shipment_weight( $weight ) { + $this->set_prop( 'label_default_shipment_weight', ( '' === $weight ? '' : wc_format_decimal( $weight ) ) ); + } + + public function set_label_minimum_shipment_weight( $weight ) { + $this->set_prop( 'label_minimum_shipment_weight', ( '' === $weight ? '' : wc_format_decimal( $weight ) ) ); + } + + public function set_label_auto_enable( $enable ) { + $this->set_prop( 'label_auto_enable', wc_string_to_bool( $enable ) ); + } + + public function set_label_auto_shipment_status_shipped( $enable ) { + $this->set_prop( 'label_auto_shipment_status_shipped', wc_string_to_bool( $enable ) ); + } + + public function set_label_auto_shipment_status( $status ) { + $this->set_prop( 'label_auto_shipment_status', $status ); + } + + public function set_label_return_auto_enable( $enable ) { + $this->set_prop( 'label_return_auto_enable', wc_string_to_bool( $enable ) ); + } + + public function set_label_return_auto_shipment_status( $status ) { + $this->set_prop( 'label_return_auto_shipment_status', $status ); + } + + public function get_label_classname( $type ) { + return '\Vendidero\Germanized\Shipments\Labels\Simple'; + } + + /** + * Whether or not this instance is a manual integration. + * Manual integrations are constructed dynamically from DB and do not support + * automatic shipment handling, e.g. label creation. + * + * @return bool + */ + public function is_manual_integration() { + return false; + } + + /** + * Whether or not this instance supports a certain label type. + * + * @param string $label_type The label type e.g. simple or return. + * + * @return bool + */ + public function supports_labels( $label_type, $shipment = false ) { + return true; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * + * @return mixed|void + */ + public function get_label( $shipment ) { + $type = wc_gzd_get_label_type_by_shipment( $shipment ); + $label = wc_gzd_get_label_by_shipment( $shipment, $type ); + + return apply_filters( "{$this->get_hook_prefix()}label", $label, $shipment, $this ); + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + public function get_label_fields_html( $shipment ) { + /** + * Setup local variables + */ + $settings = $this->get_label_fields( $shipment ); + $provider = $this; + + if ( is_wp_error( $settings ) ) { + $error = $settings; + + ob_start(); + include Package::get_path() . '/includes/admin/views/label/html-shipment-label-backbone-error.php'; + $html = ob_get_clean(); + } else { + ob_start(); + include Package::get_path() . '/includes/admin/views/label/html-shipment-label-backbone-form.php'; + $html = ob_get_clean(); + } + + return apply_filters( "{$this->get_hook_prefix()}label_fields_html", $html, $shipment, $this ); + } + + protected function get_automation_settings( $for_shipping_method = false ) { + $settings = array( + array( + 'title' => _x( 'Automation', 'shipments', 'woocommerce-germanized' ), + 'allow_override' => true, + 'type' => 'title', + 'id' => 'shipping_provider_label_auto_options', + ), + ); + + $shipment_statuses = array_diff_key( wc_gzd_get_shipment_statuses(), array_fill_keys( array( 'gzd-draft', 'gzd-delivered', 'gzd-returned', 'gzd-requested' ), '' ) ); + + $settings = array_merge( + $settings, + array( + array( + 'title' => _x( 'Labels', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Automatically create labels for shipments.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'label_auto_enable', + 'type' => 'gzd_toggle', + 'value' => wc_bool_to_string( $this->get_setting( 'label_auto_enable' ) ), + ), + + array( + 'title' => _x( 'Status', 'shipments', 'woocommerce-germanized' ), + 'type' => 'select', + 'id' => 'label_auto_shipment_status', + 'desc' => '
' . _x( 'Choose a shipment status which should trigger generation of a label.', 'shipments', 'woocommerce-germanized' ) . ' ' . ( 'yes' === Package::get_setting( 'auto_enable' ) ? sprintf( _x( 'Your current default shipment status is: %s.', 'shipments', 'woocommerce-germanized' ), wc_gzd_get_shipment_status_name( Package::get_setting( 'auto_default_status' ) ) ) : '' ) . '
', + 'options' => $shipment_statuses, + 'class' => 'wc-enhanced-select', + 'custom_attributes' => array( 'data-show_if_label_auto_enable' => '' ), + 'value' => $this->get_setting( 'label_auto_shipment_status' ), + ), + + array( + 'title' => _x( 'Shipment Status', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Mark shipment as shipped after label has been created successfully.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'label_auto_shipment_status_shipped', + 'type' => 'gzd_toggle', + 'value' => wc_bool_to_string( $this->get_setting( 'label_auto_shipment_status_shipped' ) ), + ), + ) + ); + + if ( $this->supports_labels( 'return' ) ) { + $settings = array_merge( + $settings, + array( + array( + 'title' => _x( 'Returns', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Automatically create labels for returns.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'label_return_auto_enable', + 'type' => 'gzd_toggle', + 'value' => wc_bool_to_string( $this->get_setting( 'label_return_auto_enable' ) ), + ), + + array( + 'title' => _x( 'Status', 'shipments', 'woocommerce-germanized' ), + 'type' => 'select', + 'id' => 'label_return_auto_shipment_status', + 'desc' => '
' . _x( 'Choose a shipment status which should trigger generation of a return label.', 'shipments', 'woocommerce-germanized' ) . '
', + 'options' => $shipment_statuses, + 'class' => 'wc-enhanced-select', + 'custom_attributes' => array( 'data-show_if_label_return_auto_enable' => '' ), + 'value' => $this->get_setting( 'label_return_auto_shipment_status' ), + ), + ) + ); + } + + $settings = array_merge( + $settings, + array( + array( + 'type' => 'sectionend', + 'id' => 'shipping_provider_label_auto_options', + ), + ) + ); + + return $settings; + } + + public function get_settings_help_pointers( $section = '' ) { + return array(); + } + + protected function get_label_settings( $for_shipping_method = false ) { + $settings = array( + array( + 'title' => '', + 'type' => 'title', + 'id' => 'shipping_provider_label_options', + ), + + array( + 'title' => _x( 'Default content weight (kg)', 'shipments', 'woocommerce-germanized' ), + 'type' => 'text', + 'desc' => _x( 'Choose a default shipment content weight to be used for labels if no weight has been applied to the shipment.', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => true, + 'id' => 'label_default_shipment_weight', + 'css' => 'max-width: 60px;', + 'class' => 'wc_input_decimal', + 'default' => $this->get_default_label_default_shipment_weight(), + 'value' => wc_format_localized_decimal( $this->get_setting( 'label_default_shipment_weight' ) ), + ), + + array( + 'title' => _x( 'Minimum weight (kg)', 'shipments', 'woocommerce-germanized' ), + 'type' => 'text', + 'desc' => _x( 'Choose a minimum weight to be used for labels e.g. to prevent low shipment weight errors.', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => true, + 'id' => 'label_minimum_shipment_weight', + 'css' => 'max-width: 60px;', + 'class' => 'wc_input_decimal', + 'default' => $this->get_default_label_minimum_shipment_weight(), + 'value' => wc_format_localized_decimal( $this->get_setting( 'label_minimum_shipment_weight' ) ), + ), + + array( + 'type' => 'sectionend', + 'id' => 'shipping_provider_label_options', + ), + ); + + return $settings; + } + + protected function get_available_base_countries() { + $countries = array(); + + if ( function_exists( 'WC' ) && WC()->countries ) { + $countries = WC()->countries->get_countries(); + } + + return $countries; + } + + public function get_setting_sections() { + $sections = array( + '' => _x( 'General', 'shipments', 'woocommerce-germanized' ), + 'label' => _x( 'Labels', 'shipments', 'woocommerce-germanized' ), + 'automation' => _x( 'Automation', 'shipments', 'woocommerce-germanized' ), + ); + + $sections = array_replace_recursive( $sections, parent::get_setting_sections() ); + + return $sections; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + public function get_label_fields( $shipment ) { + if ( 'return' === $shipment->get_type() ) { + return $this->get_return_label_fields( $shipment ); + } else { + return $this->get_simple_label_fields( $shipment ); + } + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + protected function get_simple_label_fields( $shipment ) { + $default = $this->get_default_label_product( $shipment ); + $available = $this->get_available_label_products( $shipment ); + + $settings = array( + array( + 'id' => 'product_id', + 'label' => sprintf( _x( '%s Product', 'shipments', 'woocommerce-germanized' ), $this->get_title() ), + 'description' => '', + 'options' => $this->get_available_label_products( $shipment ), + 'value' => $default && array_key_exists( $default, $available ) ? $default : '', + 'type' => 'select', + ), + ); + + return $settings; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + protected function get_return_label_fields( $shipment ) { + return $this->get_simple_label_fields( $shipment ); + } + + /** + * @param Shipment $shipment + * @param $props + * + * @return ShipmentError|mixed + */ + protected function validate_label_request( $shipment, $props ) { + return $props; + } + + /** + * @param Shipment $shipment + * + * @return array + */ + protected function get_default_label_props( $shipment ) { + $default = array( + 'shipping_provider' => $this->get_name(), + 'weight' => wc_gzd_get_shipment_label_weight( $shipment ), + 'net_weight' => wc_gzd_get_shipment_label_weight( $shipment, true ), + 'shipment_id' => $shipment->get_id(), + 'services' => array(), + 'product_id' => $this->get_default_label_product( $shipment ), + ); + + $dimensions = wc_gzd_get_shipment_label_dimensions( $shipment ); + $default = array_merge( $default, $dimensions ); + + return $default; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * @param mixed $props + * + * @return ShipmentError|true + */ + public function create_label( $shipment, $props = false ) { + /** + * In case props is false this indicates an automatic (non-manual) request. + */ + if ( false === $props ) { + $props = $this->get_default_label_props( $shipment ); + } elseif ( is_array( $props ) ) { + $fields = $this->get_label_fields( $shipment ); + + /** + * By default checkbox fields won't be transmitted via POST data. + * In case the values does not exist within props, assume not checked. + */ + foreach ( $fields as $field ) { + if ( ! isset( $field['value'] ) ) { + continue; + } + + if ( 'checkbox' === $field['type'] && ! isset( $props[ $field['id'] ] ) ) { + // Exclude array fields from default checkbox handling + if ( isset( $field['name'] ) && strstr( $field['name'], '[]' ) ) { + continue; + } + + $props[ $field['id'] ] = 'no'; + } elseif ( 'multiselect' === $field['type'] ) { + if ( isset( $props[ $field['id'] ] ) ) { + $props[ $field['id'] ] = (array) $props[ $field['id'] ]; + } + } + } + + /** + * Merge with default data. That needs to be done after manually + * parsing checkboxes as missing data would be overridden with defaults. + */ + $props = wp_parse_args( $props, $this->get_default_label_props( $shipment ) ); + + foreach ( $props as $key => $value ) { + if ( substr( $key, 0, strlen( 'service_' ) ) === 'service_' ) { + $new_key = substr( $key, ( strlen( 'service_' ) ) ); + + if ( wc_string_to_bool( $value ) && in_array( $new_key, $this->get_available_label_services( $shipment ), true ) ) { + if ( ! in_array( $new_key, $props['services'], true ) ) { + $props['services'][] = $new_key; + } + unset( $props[ $key ] ); + } else { + if ( ( $service_key = array_search( $new_key, $props['services'], true ) ) !== false ) { + unset( $props['services'][ $service_key ] ); + } + unset( $props[ $key ] ); + } + } + } + } + + $props = $this->validate_label_request( $shipment, $props ); + + if ( is_wp_error( $props ) ) { + return $props; + } + + if ( isset( $props['services'] ) ) { + $props['services'] = array_unique( $props['services'] ); + } + + $label = Factory::get_label( 0, $this->get_name(), $shipment->get_type() ); + + if ( $label ) { + foreach ( $props as $key => $value ) { + $setter = "set_{$key}"; + + if ( is_callable( array( $label, $setter ) ) ) { + $label->{$setter}( $value ); + } else { + $label->update_meta_data( $key, $value ); + } + } + + $label->set_shipment( $shipment ); + + /** + * Fetch the label via API and store as file + */ + $result = $label->fetch(); + + if ( is_wp_error( $result ) ) { + $result = wc_gzd_get_shipment_error( $result ); + + if ( ! $result->is_soft_error() ) { + return $result; + } + } + + do_action( "{$this->get_general_hook_prefix()}created_label", $label, $this ); + $label_id = $label->save(); + + return is_wp_error( $result ) && $result->is_soft_error() ? $result : $label_id; + } + + return new ShipmentError( 'label-error', _x( 'Error while creating the label.', 'shipments', 'woocommerce-germanized' ) ); + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + public function get_available_label_services( $shipment ) { + return array(); + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + abstract public function get_available_label_products( $shipment ); + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + abstract public function get_default_label_product( $shipment ); +} diff --git a/packages/woocommerce-germanized-shipments/src/ShippingProvider/Helper.php b/packages/woocommerce-germanized-shipments/src/ShippingProvider/Helper.php new file mode 100644 index 000000000..948ac1687 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShippingProvider/Helper.php @@ -0,0 +1,217 @@ +get_shipping_provider_class_names(); + + if ( ! is_object( $provider ) ) { + if ( ! class_exists( $provider ) ) { + return false; + } + + $provider = new $provider(); + } else { + $classname = '\Vendidero\Germanized\Shipments\ShippingProvider\Simple'; + + if ( array_key_exists( $provider->shipping_provider_name, $classes ) ) { + $classname = $classes[ $provider->shipping_provider_name ]; + } + + $classname = apply_filters( 'woocommerce_gzd_shipping_provider_class_name', $classname, $provider->shipping_provider_name, $provider ); + + if ( ! class_exists( $classname ) ) { + $classname = '\Vendidero\Germanized\Shipments\ShippingProvider\Simple'; + } + + $provider = new $classname( $provider ); + } + + if ( ! $provider || ! is_a( $provider, '\Vendidero\Germanized\Shipments\Interfaces\ShippingProvider' ) ) { + return false; + } + + if ( is_null( $this->shipping_providers ) ) { + $this->shipping_providers = array(); + } + + $this->shipping_providers[ $provider->get_name() ] = $provider; + } + + /** + * Shipping providers register themselves by returning their main class name through the woocommerce_gzd_shipping_provider_integrations filter. + * + * @return array + */ + public function get_shipping_provider_class_names() { + $class_names = array(); + + /** + * This filter may be used to register additional shipping providers + * by adding a unique name as key and the classname to be loaded as value of the array. + * + * @param array $shipping_providers The shipping provider array + * + * @since 1.0.5 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( 'woocommerce_gzd_shipping_provider_class_names', $class_names ); + } + + public function is_shipping_provider_activated( $name ) { + /** + * Make sure that the plugin has initialised, e.g. during installs of shipping provider + */ + if ( ! did_action( 'woocommerce_gzd_shipments_init' ) ) { + Package::init(); + } + + return WC_Data_Store::load( 'shipping-provider' )->is_activated( $name ); + } + + /** + * Loads all shipping providers which are hooked in. + * + * @return ShippingProvider[] + */ + public function load_shipping_providers() { + if ( ! did_action( 'plugins_loaded' ) || doing_action( 'plugins_loaded' ) ) { + wc_doing_it_wrong( __FUNCTION__, _x( 'Loading shipping providers should only be triggered after the plugins_loaded action has fully been executed', 'shipments', 'woocommerce-germanized' ), '2.2.3' ); + return array(); + } + + $this->shipping_providers = array(); + + $shipping_providers = WC_Data_Store::load( 'shipping-provider' )->get_shipping_providers(); + $registered_providers = $this->get_shipping_provider_class_names(); + + foreach ( $registered_providers as $k => $provider ) { + if ( ! array_key_exists( $k, $shipping_providers ) ) { + $shipping_providers[ $k ] = $provider; + } + } + + // For the settings in the backend, and for non-shipping zone methods, we still need to load any registered classes here. + foreach ( $shipping_providers as $provider_name => $provider_class ) { + $this->register_shipping_provider( $provider_class ); + } + + /** + * This hook fires as soon as shipping providers are loaded. + * Additional shipping provider may be registered manually afterwards. + * + * @param Helper $providers The shipping providers instance + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_load_shipping_providers', $this ); + + // Return loaded methods. + return $this->get_shipping_providers(); + } + + /** + * Returns all registered shipping providers for usage. + * + * @return Simple|Auto|ShippingProvider[] + */ + public function get_shipping_providers() { + if ( is_null( $this->shipping_providers ) ) { + $this->load_shipping_providers(); + } + + if ( is_null( $this->shipping_providers ) ) { + return array(); + } + + return $this->shipping_providers; + } + + /** + * @param $name + * + * @return false|Simple|Auto|ShippingProvider + */ + public function get_shipping_provider( $name ) { + $providers = $this->get_shipping_providers(); + + return ( array_key_exists( $name, $providers ) ? $providers[ $name ] : false ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ShippingProvider/Method.php b/packages/woocommerce-germanized-shipments/src/ShippingProvider/Method.php new file mode 100644 index 000000000..4155abc8b --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShippingProvider/Method.php @@ -0,0 +1,413 @@ +method = $method; + $this->init(); + } else { + $this->is_placeholder = true; + $this->init_placeholder( $method ); + } + } + + protected function init_placeholder( $id ) { + if ( is_a( $id, 'WC_Shipping_Rate' ) ) { + $instance_id = $id->get_instance_id(); + $id = $id->get_id(); + + if ( strpos( $id, ':' ) === false ) { + $id = $id . ':' . $instance_id; + } + } elseif ( is_a( $id, 'WC_Shipping_Method' ) ) { + $instance_id = $id->get_instance_id(); + $id = $id->id; + + if ( strpos( $id, ':' ) === false ) { + $id = $id . ':' . $instance_id; + } + } + + if ( ! is_numeric( $id ) ) { + $expl = explode( ':', $id ); + $instance_id = ( ( ! empty( $expl ) && count( $expl ) > 1 ) ? $expl[1] : 0 ); + $id = ( ( ! empty( $expl ) && count( $expl ) > 1 ) ? $expl[0] : $id ); + } else { + $instance_id = $id; + } + + $this->placeholder_id = $id; + $this->placeholder_instance_id = $instance_id; + + $this->instance_form_fields = Package::get_method_settings(); + } + + public function get_fallback_setting_value( $setting_key ) { + $setting_key = $this->maybe_prefix_key( $setting_key ); + $setting_value = ''; + + /** + * In case the setting belongs to the current shipping provider + * lets allow overriding the fallback setting with data from the provider. + */ + if ( ( $provider = $this->get_provider_instance() ) && $this->setting_belongs_to_provider( $setting_key ) ) { + $setting_value = $provider->get_setting( $setting_key ); + } + + if ( is_null( $setting_value ) ) { + $setting_value = Package::get_setting( $setting_key, null ); + } + + /** + * Convert booleans to string options + */ + if ( is_bool( $setting_value ) ) { + $setting_value = wc_bool_to_string( $setting_value ); + } + + return apply_filters( "{$this->get_hook_prefix()}setting_fallback_value", $setting_value, $setting_key, $this ); + } + + protected function supports_instance_settings() { + if ( $this->is_placeholder() ) { + return false; + } else { + $supports_settings = ( $this->method->supports( 'instance-settings' ) ) ? true : false; + + return apply_filters( 'woocommerce_gzd_shipping_provider_method_supports_instance_settings', $supports_settings, $this ); + } + } + + public function is_placeholder() { + return true === $this->is_placeholder; + } + + /** + * Get all available shipping method settings. This method (re-) loads all + * the settings available across every registered shipping provider. + * Call the cached version instead for performance improvements. + * + * @see Package::get_method_settings() + * + * @return mixed|void + */ + public static function get_admin_settings() { + /** + * Filter to adjust admin settings added to the shipment method instance specifically for shipping providers. + * + * @param array $settings Admin setting fields. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + $settings = apply_filters( + 'woocommerce_gzd_shipping_provider_method_admin_settings', + array( + 'shipping_provider_title' => array( + 'title' => _x( 'Shipping Provider Settings', 'shipments', 'woocommerce-germanized' ), + 'type' => 'title', + 'default' => '', + 'description' => _x( 'Adjust shipping provider settings used for managing shipments.', 'shipments', 'woocommerce-germanized' ), + ), + 'shipping_provider' => array( + 'title' => _x( 'Shipping Provider', 'shipments', 'woocommerce-germanized' ), + 'type' => 'select', + /** + * Filter to adjust default shipping provider pre-selected within shipping provider method settings. + * + * @param string $provider_name The shipping provider name e.g. dhl. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + 'default' => apply_filters( 'woocommerce_gzd_shipping_provider_method_default_provider', '' ), + 'options' => wc_gzd_get_shipping_provider_select(), + 'description' => _x( 'Choose a shipping provider which will be selected by default for an eligible shipment.', 'shipments', 'woocommerce-germanized' ), + ), + ) + ); + + foreach ( wc_gzd_get_shipping_providers() as $provider ) { + if ( ! $provider->is_activated() ) { + continue; + } + + $additional_settings = $provider->get_shipping_method_settings(); + $settings = array_merge( $settings, $additional_settings ); + } + + /** + * Append a stop title to make sure the table is closed within settings. + */ + $settings = array_merge( + $settings, + array( + 'shipping_provider_stop_title' => array( + 'title' => '', + 'type' => 'title', + 'default' => '', + ), + ) + ); + + return apply_filters( 'woocommerce_gzd_shipping_provider_method_admin_settings_wrapped', $settings ); + } + + protected function init() { + $this->instance_form_fields = Package::get_method_settings(); + + if ( ! array_key_exists( 'shipping_provider', $this->get_method()->instance_form_fields ) ) { + $this->get_method()->instance_form_fields = array_merge( $this->get_method()->instance_form_fields, $this->instance_form_fields ); + } + + // Refresh instance settings in case they were already loaded + if ( ! empty( $this->get_method()->instance_settings ) ) { + $this->get_method()->init_instance_settings(); + } + } + + protected function get_hook_prefix() { + $prefix = 'woocommerce_gzd_shipping_provider_method_'; + + return $prefix; + } + + /** + * Returns the Woo WC_Shipping_Method original object + * + * @return object|WC_Shipping_Method + */ + public function get_method() { + return $this->method; + } + + public function get_id() { + if ( ! $this->is_placeholder() ) { + return $this->method->id; + } else { + return $this->placeholder_id; + } + } + + public function get_instance_id() { + if ( ! $this->is_placeholder() ) { + return $this->method->get_instance_id(); + } else { + return $this->placeholder_instance_id; + } + } + + public function has_option( $key ) { + $fields = $this->instance_form_fields; + $key = $this->maybe_prefix_key( $key ); + $has_option = ( array_key_exists( $key, $fields ) && $this->setting_belongs_to_provider( $key ) ) ? true : false; + + /** + * Filter that allows checking whether a shipping provider method has a specific option or not. + * + * @param boolean $has_option Whether or not the option exists. + * @param string $key The setting key. + * @param Method $method The method instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}setting_prefix", $has_option, $key, $this ); + } + + public function setting_belongs_to_provider( $setting_key, $provider = '' ) { + $prefix = $this->get_custom_setting_prefix_key(); + + if ( ! empty( $provider ) ) { + $prefix = $provider . '_'; + } + + $belongs_to_provider = false; + + if ( ! empty( $prefix ) && substr( $setting_key, 0, strlen( $prefix ) ) === $prefix ) { + $belongs_to_provider = true; + } + + return $belongs_to_provider; + } + + public function is_provider_enabled( $provider ) { + return ( $this->get_provider() === $provider ) ? true : false; + } + + public function set_provider( $provider_name ) { + $this->provider_slug = $provider_name; + } + + public function get_provider() { + $id = sanitize_key( $this->get_id() ); + + if ( is_null( $this->provider_slug ) ) { + $provider_slug = $this->method ? $this->method->get_option( 'shipping_provider' ) : ''; + + if ( ! empty( $provider_slug ) ) { + if ( $provider = wc_gzd_get_shipping_provider( $provider_slug ) ) { + if ( ! $provider->is_activated() ) { + $provider_slug = ''; + } + } + } + + if ( empty( $provider_slug ) ) { + $provider_slug = wc_gzd_get_default_shipping_provider(); + } + + /** + * Filter that allows adjusting the shipping provider chosen for a specific shipping method. + * + * @param string $provider_slug The shipping provider. + * @param string $method_id The shipping method id. + * @param Method $method The method instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + $this->provider_slug = apply_filters( 'woocommerce_gzd_shipping_provider_method_provider', $provider_slug, $this->get_id(), $this ); + } + + /** + * Filter that allows choosing a shipping provider for a specific shipping method. + * + * The dynamic portion of this hook, `$id` refers to the shipping method id. + * + * Example hook name: `woocommerce_gzd_shipping_provider_method_flat_rate_provider` + * + * @param string $provider_slug The shipping provider name to be used. + * @param Method $method The method instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}{$id}_provider", $this->provider_slug, $this ); + } + + public function get_provider_instance() { + $provider_slug = $this->get_provider(); + + if ( ! empty( $provider_slug ) ) { + return wc_gzd_get_shipping_provider( $provider_slug ); + } + + return false; + } + + protected function get_custom_setting_prefix_key() { + $prefix = ''; + + if ( $provider = $this->get_provider_instance() ) { + $prefix = $provider->get_name() . '_'; + } + + return apply_filters( "{$this->get_hook_prefix()}custom_setting_prefix", $prefix, $this ); + } + + protected function maybe_prefix_key( $key ) { + $fields = $this->instance_form_fields; + $prefix = $this->get_custom_setting_prefix_key(); + $new_key = $key; + + // Do only prefix if the prefix does not yet exist. + if ( ! array_key_exists( $new_key, $fields ) ) { + if ( substr( $key, 0, strlen( $prefix ) ) !== $prefix ) { + $new_key = $prefix . $key; + } + } + + /** + * Filter that allows prefixing the setting key used for a shipping provider method. + * + * @param string $new_key The prefixed setting key. + * @param string $key The original setting key. + * @param Method $method The method instance. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}setting_key_prefixed", $new_key, $key, $this ); + } + + public function get_option( $key ) { + $key = $this->maybe_prefix_key( $key ); + $option_value = $this->get_fallback_setting_value( $key ); + + if ( ! $this->is_placeholder() ) { + if ( $this->has_option( $key ) && $this->supports_instance_settings() ) { + $option_type = isset( $this->instance_form_fields[ $key ]['type'] ) ? $this->instance_form_fields[ $key ]['type'] : 'text'; + + // Do only use method settings if the method is not a placeholder and method supports settings + $option_value = $this->method->get_option( $key, $option_value ); + + if ( in_array( $option_type, array( 'checkbox', 'radio' ), true ) ) { + $option_value = wc_string_to_bool( $option_value ); + + if ( $option_value ) { + $option_value = 'yes'; + } else { + $option_value = 'no'; + } + } + } + } + + return apply_filters( "{$this->get_hook_prefix()}setting_value", $option_value, $key, $this ); + } + + /** + * Call child methods if the method does not exist. + * + * @param $method + * @param $args + * + * @return bool|mixed + */ + public function __call( $method, $args ) { + + if ( method_exists( $this->method, $method ) ) { + return call_user_func_array( array( $this->method, $method ), $args ); + } + + return false; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/ShippingProvider/MethodPlaceholder.php b/packages/woocommerce-germanized-shipments/src/ShippingProvider/MethodPlaceholder.php new file mode 100644 index 000000000..bd4d02029 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/ShippingProvider/MethodPlaceholder.php @@ -0,0 +1,24 @@ + true, + 'title' => '', + 'name' => '', + 'description' => '', + 'order' => 0, + 'supports_customer_returns' => false, + 'supports_guest_returns' => false, + 'return_manual_confirmation' => true, + 'return_instructions' => '', + 'tracking_url_placeholder' => '', + 'tracking_desc_placeholder' => '', + ); + + protected $address_data = array( + 'shipper' => null, + 'return' => null, + ); + + /** + * Get the provider if ID is passed. In case it is an integration, data will be provided through the impl. + * This class should NOT be instantiated, but the `wc_gzd_get_shipping_provider` function should be used. + * + * @param int|object|ShippingProvider $provider Provider to read. + */ + public function __construct( $data = 0 ) { + parent::__construct( $data ); + + if ( $data instanceof ShippingProvider ) { + $this->set_id( absint( $data->get_id() ) ); + } elseif ( is_numeric( $data ) ) { + $this->set_id( $data ); + } elseif ( is_object( $data ) && isset( $data->shipping_provider_id ) ) { + $this->set_id( $data->shipping_provider_id ); + } + + $this->data_store = WC_Data_Store::load( $this->data_store_name ); + + // If we have an ID, load the user from the DB. + if ( $this->get_id() ) { + try { + $this->data_store->read( $this ); + } catch ( Exception $e ) { + $this->set_id( 0 ); + $this->set_object_read( true ); + } + } else { + $this->set_object_read( true ); + } + } + + public function get_help_link() { + return ''; + } + + public function get_signup_link() { + return ''; + } + + public function is_pro() { + return false; + } + + /** + * Whether or not this instance is a manual integration. + * Manual integrations are constructed dynamically from DB and do not support + * automatic shipment handling, e.g. label creation. + * + * @return bool + */ + public function is_manual_integration() { + return true; + } + + /** + * Whether or not this instance supports a certain label type. + * + * @param string $label_type The label type e.g. simple or return. + * @param false|Shipment Shipment instance + * + * @return bool + */ + public function supports_labels( $label_type, $shipment = false ) { + return false; + } + + public function supports_customer_return_requests() { + if ( $this->is_manual_integration() ) { + return true; + } + + return false; + } + + /** + * Some providers (e.g. DHL) create return labels automatically and the return + * address is chosen dynamically depending on the country. For that reason the return address + * might not show up within emails or in customer panel. + * + * @return bool + */ + public function hide_return_address() { + return false; + } + + public function get_edit_link( $section = '' ) { + $url = admin_url( 'admin.php?page=wc-settings&tab=germanized-shipping_provider&provider=' . esc_attr( $this->get_name() ) ); + $url = add_query_arg( array( 'section' => $section ), $url ); + + return esc_url_raw( $url ); + } + + /** + * Returns whether the shipping provider is active for usage or not. + * + * @return bool + */ + public function is_activated() { + return $this->get_activated() === true; + } + + public function needs_manual_confirmation_for_returns() { + return $this->get_return_manual_confirmation() === true; + } + + /** + * @param false|\WC_Order $order + * + * @return bool + */ + public function supports_customer_returns( $order = false ) { + return $this->get_supports_customer_returns() === true; + } + + public function supports_guest_returns() { + return $this->get_supports_customer_returns() === true && $this->get_supports_guest_returns() === true; + } + + /** + * Returns a title for the shipping provider. + * + * @param string $context + * + * @return string + */ + public function get_title( $context = 'view' ) { + return $this->get_prop( 'title', $context ); + } + + /** + * Returns the provider order. + * + * @param string $context + * + * @return int + */ + public function get_order( $context = 'view' ) { + return $this->get_prop( 'order', $context ); + } + + /** + * Returns a unique slug/name for the shipping provider. + * + * @param string $context + * + * @return string + */ + public function get_name( $context = 'view' ) { + return $this->get_prop( 'name', $context ); + } + + /** + * Returns a description for the provider. + * + * @param string $context + * + * @return string + */ + public function get_description( $context = 'view' ) { + $desc = $this->get_prop( 'description', $context ); + + if ( 'view' === $context && empty( $desc ) ) { + return '-'; + } + + return $desc; + } + + /** + * Returns whether the shipping provider is activated or not. + * + * @param string $context + * + * @return string + */ + public function get_activated( $context = 'view' ) { + return $this->get_prop( 'activated', $context ); + } + + /** + * Returns whether the shipping provider needs manual confirmation for a return. + * + * @param string $context + * + * @return string + */ + public function get_return_manual_confirmation( $context = 'view' ) { + return $this->get_prop( 'return_manual_confirmation', $context ); + } + + /** + * Returns whether the shipping provider supports returns added by customers or not. + * + * @param string $context + * + * @return string + */ + public function get_supports_customer_returns( $context = 'view' ) { + return $this->get_prop( 'supports_customer_returns', $context ); + } + + /** + * Returns whether the shipping provider supports returns added by guests or not. + * + * @param string $context + * + * @return string + */ + public function get_supports_guest_returns( $context = 'view' ) { + return $this->get_prop( 'supports_guest_returns', $context ); + } + + /** + * Returns the tracking url placeholder which is being used to + * construct a tracking url. + * + * @param string $context + * + * @return mixed + */ + public function get_tracking_url_placeholder( $context = 'view' ) { + $data = $this->get_prop( 'tracking_url_placeholder', $context ); + + // In case the option value is not stored in DB yet + if ( 'view' === $context && empty( $data ) ) { + $data = $this->get_default_tracking_url_placeholder(); + } + + return $data; + } + + public function get_default_tracking_url_placeholder() { + return ''; + } + + /** + * Returns the tracking description placeholder which is being used to + * construct a tracking description. + * + * @param string $context + * + * @return mixed + */ + public function get_tracking_desc_placeholder( $context = 'view' ) { + $data = $this->get_prop( 'tracking_desc_placeholder', $context ); + + // In case the option value is not stored in DB yet + if ( 'view' === $context && empty( $data ) ) { + $data = $this->get_default_tracking_desc_placeholder(); + } + + return $data; + } + + public function get_default_tracking_desc_placeholder() { + return _x( 'Your shipment is being processed by {shipping_provider}. If you want to track the shipment, please use the following tracking number: {tracking_id}. Depending on the chosen shipping method it is possible that the tracking data does not reflect the current status when receiving this email.', 'shipments', 'woocommerce-germanized' ); + } + + /** + * Returns the return instructions. + * + * @param string $context + * + * @return mixed + */ + public function get_return_instructions( $context = 'view' ) { + return $this->get_prop( 'return_instructions', $context ); + } + + public function has_return_instructions() { + $instructions = $this->get_return_instructions(); + + return empty( $instructions ) ? false : true; + } + + protected function get_address_props( $address_type = 'shipper' ) { + if ( is_null( $this->address_data[ $address_type ] ) ) { + $this->address_data[ $address_type ] = wc_gzd_get_shipment_setting_address_fields( $address_type ); + } + + return $this->address_data[ $address_type ]; + } + + public function get_shipper_address_data() { + return $this->get_address_props( 'shipper' ); + } + + public function get_address_prop( $prop, $type = 'shipper' ) { + $address_fields = $this->get_address_props( $type ); + + return array_key_exists( $prop, $address_fields ) ? $address_fields[ $prop ] : ''; + } + + public function get_shipper_email() { + return $this->get_address_prop( 'email' ); + } + + public function get_shipper_phone() { + return $this->get_address_prop( 'phone' ); + } + + public function get_contact_phone() { + return get_option( 'woocommerce_gzd_shipments_contact_phone' ); + } + + public function get_shipper_first_name() { + return $this->get_address_prop( 'first_name' ); + } + + public function get_shipper_last_name() { + return $this->get_address_prop( 'last_name' ); + } + + public function get_shipper_name() { + return $this->get_shipper_formatted_full_name(); + } + + public function get_shipper_formatted_full_name() { + return $this->get_address_prop( 'full_name' ); + } + + public function get_shipper_company() { + return $this->get_address_prop( 'company' ); + } + + public function get_shipper_address() { + return $this->get_address_prop( 'address_1' ); + } + + public function get_shipper_address_1() { + return $this->get_shipper_address(); + } + + public function get_shipper_address_2() { + return $this->get_address_prop( 'address_2' ); + } + + public function get_shipper_street() { + return $this->get_address_prop( 'street' ); + } + + public function get_shipper_street_number() { + return $this->get_address_prop( 'street_number' ); + } + + public function get_shipper_postcode() { + return $this->get_address_prop( 'postcode' ); + } + + public function get_shipper_city() { + return $this->get_address_prop( 'city' ); + } + + public function get_shipper_customs_reference_number() { + return $this->get_address_prop( 'customs_reference_number' ); + } + + public function get_shipper_customs_uk_vat_id() { + return $this->get_address_prop( 'customs_uk_vat_id' ); + } + + public function get_shipper_country() { + $country_data = wc_format_country_state_string( $this->get_address_prop( 'country' ) ); + + return $country_data['country']; + } + + public function get_shipper_state() { + $country_data = wc_format_country_state_string( $this->get_address_prop( 'country' ) ); + + return $country_data['state']; + } + + public function get_return_address_data() { + return $this->get_address_props( 'return' ); + } + + public function get_return_first_name() { + return $this->get_address_prop( 'first_name', 'return' ); + } + + public function get_return_last_name() { + return $this->get_address_prop( 'last_name', 'return' ); + } + + public function get_return_company() { + return $this->get_address_prop( 'company', 'return' ); + } + + public function get_return_name() { + return $this->get_return_formatted_full_name(); + } + + public function get_return_formatted_full_name() { + return $this->get_address_prop( 'full_name', 'return' ); + } + + public function get_return_address() { + return $this->get_address_prop( 'address_1', 'return' ); + } + + public function get_return_address_2() { + return $this->get_address_prop( 'address_2', 'return' ); + } + + public function get_return_street() { + return $this->get_address_prop( 'street', 'return' ); + } + + public function get_return_street_number() { + return $this->get_address_prop( 'street_number', 'return' ); + } + + public function get_return_postcode() { + return $this->get_address_prop( 'postcode', 'return' ); + } + + public function get_return_city() { + return $this->get_address_prop( 'city', 'return' ); + } + + public function get_return_country() { + $country_data = wc_format_country_state_string( $this->get_address_prop( 'country', 'return' ) ); + + return $country_data['country']; + } + + public function get_return_state() { + $country_data = wc_format_country_state_string( $this->get_address_prop( 'country', 'return' ) ); + + return $country_data['state']; + } + + public function get_return_email() { + return $this->get_address_prop( 'email', 'return' ); + } + + public function get_return_phone() { + return $this->get_address_prop( 'phone', 'return' ); + } + + /** + * Set the current shipping provider to active or inactive. + * + * @param bool $is_activated + */ + public function set_activated( $is_activated ) { + $this->set_prop( 'activated', wc_string_to_bool( $is_activated ) ); + } + + /** + * Mark the current shipping provider as manual needed confirmation for returns. + * + * @param bool $needs_confirmation + */ + public function set_return_manual_confirmation( $needs_confirmation ) { + $this->set_prop( 'return_manual_confirmation', wc_string_to_bool( $needs_confirmation ) ); + } + + /** + * Set whether or not the current shipping provider supports customer returns + * + * @param bool $supports + */ + public function set_supports_customer_returns( $supports ) { + $this->set_prop( 'supports_customer_returns', wc_string_to_bool( $supports ) ); + } + + /** + * Set whether or not the current shipping provider supports guest returns + * + * @param bool $supports + */ + public function set_supports_guest_returns( $supports ) { + $this->set_prop( 'supports_guest_returns', wc_string_to_bool( $supports ) ); + } + + public function update_settings_with_defaults() { + foreach ( $this->get_all_settings() as $section => $settings ) { + foreach ( $settings as $setting ) { + $type = isset( $setting['type'] ) ? $setting['type'] : 'title'; + $default = isset( $setting['default'] ) ? $setting['default'] : null; + + if ( in_array( $type, array( 'title', 'sectionend', 'html' ), true ) || ! isset( $setting['id'] ) || empty( $setting['id'] ) ) { + continue; + } + + $current_value = $this->get_setting( $setting['id'], null, 'edit' ); + + /** + * Update meta data with default value in case it does not yet exist. + */ + if ( is_null( $current_value ) && ! is_null( $default ) ) { + $this->update_setting( $setting['id'], $default ); + } + } + } + } + + /** + * Activate current ShippingProvider instance. + */ + public function activate() { + $this->set_activated( true ); + $this->update_settings_with_defaults(); + $this->save(); + + /** + * This action fires as soon as a certain shipping provider gets activated. + * + * @param ShippingProvider $shipping_provider The shipping provider instance. + */ + do_action( 'woocommerce_gzd_shipping_provider_activated', $this ); + } + + /** + * Deactivate current ShippingProvider instance. + */ + public function deactivate() { + $this->set_activated( false ); + $this->save(); + + /** + * This action fires as soon as a certain shipping provider gets deactivated. + * + * @param ShippingProvider $shipping_provider The shipping provider instance. + */ + do_action( 'woocommerce_gzd_shipping_provider_deactivated', $this ); + } + + /** + * Set the name of the current shipping provider. + * + * @param string $name + */ + public function set_name( $name ) { + $this->set_prop( 'name', $name ); + } + + /** + * Set the title of the current shipping provider. + * + * @param string $title + */ + public function set_title( $title ) { + $this->set_prop( 'title', $title ); + } + + /** + * Set the order of the current shipping provider. + * + * @param int $order + */ + public function set_order( $order ) { + $this->set_prop( 'order', absint( $order ) ); + } + + /** + * Set the description of the current shipping provider. + * + * @param string $description + */ + public function set_description( $description ) { + $this->set_prop( 'description', $description ); + } + + /** + * Set the return instructions of the current shipping provider. + * + * @param string $instructions + */ + public function set_return_instructions( $instructions ) { + $this->set_prop( 'return_instructions', $instructions ); + } + + /** + * Set the tracking url placeholder of the current shipping provider. + * + * @param string $placeholder + */ + public function set_tracking_url_placeholder( $placeholder ) { + $this->set_prop( 'tracking_url_placeholder', $placeholder ); + } + + /** + * Set the tracking description placeholder of the current shipping provider. + * + * @param string $placeholder + */ + public function set_tracking_desc_placeholder( $placeholder ) { + $this->set_prop( 'tracking_desc_placeholder', $placeholder ); + } + + /** + * Returns the tracking url for a specific shipment. + * + * @param Shipment $shipment + * + * @return string + */ + public function get_tracking_url( $shipment ) { + + $tracking_url = ''; + $tracking_id = $shipment->get_tracking_id(); + + if ( '' !== $this->get_tracking_url_placeholder() && ! empty( $tracking_id ) ) { + $placeholders = $this->get_tracking_placeholders( $shipment ); + $tracking_url = str_replace( array_keys( $placeholders ), array_values( $placeholders ), $this->get_tracking_url_placeholder() ); + } + + /** + * This filter returns the tracking url provided by the shipping provider for a certain shipment. + * + * The dynamic portion of the hook `$this->get_hook_prefix()` refers to the + * current provider name. + * + * Example hook name: woocommerce_gzd_shipping_provider_dhl_get_tracking_url + * + * @param string $tracking_url The tracking url. + * @param Shipment $shipment The shipment used to build the url. + * @param ShippingProvider $provider The shipping provider. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( $this->get_hook_prefix() . 'tracking_url', $tracking_url, $shipment, $this ); + } + + /** + * Returns the tracking description for a certain shipment. + * + * @param Shipment $shipment + * + * @return string + */ + public function get_tracking_desc( $shipment, $plain = false ) { + $tracking_desc = ''; + $tracking_id = $shipment->get_tracking_id(); + + if ( '' !== $this->get_tracking_desc_placeholder() && ! empty( $tracking_id ) ) { + $placeholders = $this->get_tracking_placeholders( $shipment ); + + if ( ! $plain && apply_filters( "{$this->get_general_hook_prefix()}tracking_id_with_link", true, $shipment ) && $shipment->has_tracking() ) { + $placeholders['{tracking_id}'] = '' . $shipment->get_tracking_id() . ''; + } + + $tracking_desc = str_replace( array_keys( $placeholders ), array_values( $placeholders ), $this->get_tracking_desc_placeholder() ); + } + + /** + * This filter returns the tracking description provided by the shipping provider for a certain shipment. + * + * The dynamic portion of the hook `$this->get_hook_prefix()` refers to the + * current provider name. + * + * Example hook name: woocommerce_gzd_shipping_provider_dhl_get_tracking_description + * + * @param string $tracking_url The tracking description. + * @param Shipment $shipment The shipment used to build the url. + * @param ShippingProvider $provider The shipping provider. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( $this->get_hook_prefix() . 'tracking_desc', $tracking_desc, $shipment, $this ); + } + + /** + * @param bool|Shipment $shipment + * + * @return array + */ + public function get_tracking_placeholders( $shipment = false ) { + $label = false; + + if ( $shipment ) { + $label = $shipment->get_label(); + } + + /** + * This filter may be used to add or manipulate tracking placeholder data + * for a certain shipping provider. + * + * The dynamic portion of the hook `$this->get_hook_prefix()` refers to the + * current provider name. + * + * Example hook name: woocommerce_gzd_shipping_provider_dhl_get_tracking_placeholders + * + * @param array $placeholders Placeholders in key => value pairs. + * @param ShippingProvider $provider The shipping provider. + * @param Shipment|bool $shipment The shipment instance if available. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( + "{$this->get_hook_prefix()}tracking_placeholders", + array( + '{shipment_number}' => $shipment ? $shipment->get_shipment_number() : '', + '{order_number}' => $shipment ? $shipment->get_order_number() : '', + '{tracking_id}' => $shipment ? $shipment->get_tracking_id() : '', + '{postcode}' => $shipment ? $shipment->get_postcode() : '', + '{date_sent_day}' => $shipment && $shipment->get_date_sent() ? $shipment->get_date_sent()->format( 'd' ) : '', + '{date_sent_month}' => $shipment && $shipment->get_date_sent() ? $shipment->get_date_sent()->format( 'm' ) : '', + '{date_sent_year}' => $shipment && $shipment->get_date_sent() ? $shipment->get_date_sent()->format( 'Y' ) : '', + '{date_day}' => $shipment && $shipment->get_date_created() ? $shipment->get_date_created()->format( 'd' ) : '', + '{date_month}' => $shipment && $shipment->get_date_created() ? $shipment->get_date_created()->format( 'm' ) : '', + '{date_year}' => $shipment && $shipment->get_date_created() ? $shipment->get_date_created()->format( 'Y' ) : '', + '{label_date_day}' => $label ? $label->get_date_created()->format( 'd' ) : '', + '{label_date_month}' => $label ? $label->get_date_created()->format( 'm' ) : '', + '{label_date_year}' => $label ? $label->get_date_created()->format( 'Y' ) : '', + '{shipping_provider}' => $this->get_title(), + ), + $this, + $shipment + ); + } + + /** + * Prefix for action and filter hooks on data. + * + * @since 3.0.0 + * @return string + */ + protected function get_hook_prefix() { + return $this->get_general_hook_prefix() . 'get_'; + } + + /** + * Prefix for action and filter hooks on data. + * + * @since 3.0.0 + * @return string + */ + protected function get_general_hook_prefix() { + $name = sanitize_key( $this->get_name( 'edit' ) ); + + if ( empty( $name ) ) { + return 'woocommerce_gzd_shipping_provider_'; + } else { + return "woocommerce_gzd_shipping_provider_{$name}_"; + } + } + + protected function get_general_settings( $for_shipping_method = false ) { + $settings = array( + array( + 'title' => '', + 'type' => 'title', + 'id' => 'shipping_provider_options', + ), + ); + + if ( $this->is_manual_integration() ) { + $settings = array_merge( + $settings, + array( + array( + 'title' => _x( 'Title', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Choose a title for the shipping provider.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'shipping_provider_title', + 'value' => $this->get_title( 'edit' ), + 'default' => '', + 'type' => 'text', + ), + + array( + 'title' => _x( 'Description', 'shipments', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Choose a description for the shipping provider.', 'shipments', 'woocommerce-germanized' ), + 'id' => 'shipping_provider_description', + 'value' => $this->get_description( 'edit' ), + 'default' => '', + 'type' => 'textarea', + 'css' => 'width: 100%;', + ), + ) + ); + } + + $settings = array_merge( + $settings, + array( + array( + 'title' => _x( 'Tracking URL', 'shipments', 'woocommerce-germanized' ), + 'desc' => '
' . sprintf( _x( 'Adjust the placeholder used to construct the tracking URL for this shipping provider. You may use on of the following placeholders to insert the tracking id or other dynamic data: %s', 'shipments', 'woocommerce-germanized' ), '' . implode( ', ', array_keys( $this->get_tracking_placeholders() ) ) . '' ) . '
', + 'id' => 'shipping_provider_tracking_url_placeholder', + 'placeholder' => $this->get_default_tracking_url_placeholder(), + 'value' => $this->get_tracking_url_placeholder( 'edit' ), + 'default' => $this->get_default_tracking_url_placeholder(), + 'type' => 'text', + 'css' => 'width: 100%;', + ), + + array( + 'title' => _x( 'Tracking description', 'shipments', 'woocommerce-germanized' ), + 'desc' => '
' . sprintf( _x( 'Adjust the placeholder used to construct the tracking description for this shipping provider (e.g. used within notification emails). You may use on of the following placeholders to insert the tracking id or other dynamic data: %s', 'shipments', 'woocommerce-germanized' ), '' . implode( ', ', array_keys( $this->get_tracking_placeholders() ) ) . '' ) . '
', + 'id' => 'shipping_provider_tracking_desc_placeholder', + 'placeholder' => $this->get_default_tracking_desc_placeholder(), + 'value' => $this->get_tracking_desc_placeholder( 'edit' ), + 'default' => $this->get_default_tracking_desc_placeholder(), + 'type' => 'textarea', + 'css' => 'width: 100%; min-height: 60px; margin-top: 1em;', + ), + ) + ); + + $settings = array_merge( + $settings, + array( + array( + 'type' => 'sectionend', + 'id' => 'shipping_provider_options', + ), + ) + ); + + return $settings; + } + + /** + * @param Shipment $shipment + * @param $key + */ + public function get_shipment_setting( $shipment, $key, $default = null ) { + $value = $this->get_setting( $key, $default ); + + if ( $method = $shipment->get_shipping_method_instance() ) { + $prefixed_key = $this->get_name() . '_' . $key; + + /** + * Do only allow overriding settings in case the shipping provider + * selected for the shipping method matches the current shipping provider. + */ + if ( $method->get_provider() === $this->get_name() && $method->has_option( $prefixed_key ) ) { + $method_value = $method->get_option( $prefixed_key ); + + if ( ! is_null( $method_value ) && $value !== $method_value ) { + $value = $method_value; + } + } + } + + return $value; + } + + public function get_setting( $key, $default = null, $context = 'view' ) { + $clean_key = $this->unprefix_setting_key( $key ); + $getter = "get_{$clean_key}"; + $value = $default; + + if ( is_callable( array( $this, $getter ) ) ) { + $value = $this->$getter( $context ); + } elseif ( $this->meta_exists( $clean_key ) ) { + $value = $this->get_meta( $clean_key, true, $context ); + } + + if ( strstr( $key, 'password' ) ) { + if ( class_exists( 'WC_GZD_Secret_Box_Helper' ) ) { + $result = \WC_GZD_Secret_Box_Helper::decrypt( $value ); + + if ( ! is_wp_error( $result ) ) { + $value = $result; + } + } + + $value = $this->retrieve_password( $value ); + } + + return $value; + } + + protected function retrieve_password( $value ) { + return stripslashes( $value ); + } + + protected function unprefix_setting_key( $key ) { + $prefixes = array( + 'shipping_provider_', + $this->get_name() . '_', + ); + + foreach ( $prefixes as $prefix ) { + if ( substr( $key, 0, strlen( $prefix ) ) === $prefix ) { + $key = substr( $key, strlen( $prefix ) ); + } + } + + return $key; + } + + public function update_settings( $section = '', $data = null, $save = true ) { + $settings_to_save = Settings::get_sanitized_settings( $this->get_settings( $section ), $data ); + + foreach ( $settings_to_save as $option_name => $value ) { + $this->update_setting( $option_name, $value ); + } + + if ( $save ) { + $this->save(); + } + } + + public function update_setting( $setting, $value ) { + $setting_name_clean = $this->unprefix_setting_key( $setting ); + $setter = 'set_' . $setting_name_clean; + + try { + if ( is_callable( array( $this, $setter ) ) ) { + $this->{$setter}( $value ); + } else { + $this->update_meta_data( $setting_name_clean, $value ); + } + } catch ( Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } + } + + public function get_settings( $section = '', $for_shipping_method = false ) { + $settings = array(); + + if ( '' === $section || 'general' === $section ) { + $settings = $this->get_general_settings( $for_shipping_method ); + } elseif ( 'returns' === $section ) { + $settings = $this->get_return_settings( $for_shipping_method ); + } elseif ( is_callable( array( $this, "get_{$section}_settings" ) ) ) { + $settings = $this->{"get_{$section}_settings"}( $for_shipping_method ); + } + + /** + * This filter returns the admin settings available for a certain shipping provider. + * + * The dynamic portion of the hook `$this->get_hook_prefix()` refers to the + * current provider name. + * + * Example hook name: woocommerce_gzd_shipping_provider_dhl_get_settings + * + * @param array $settings Available settings. + * @param ShippingProvider $provider The shipping provider. + * + * @since 3.0.6 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( $this->get_hook_prefix() . 'settings', $settings, $section, $this, $for_shipping_method ); + } + + protected function get_return_settings( $for_shipping_method = false ) { + $settings = array( + array( + 'title' => '', + 'type' => 'title', + 'id' => 'shipping_provider_return_options', + ), + ); + + $settings = array_merge( + $settings, + array( + array( + 'title' => _x( 'Customer returns', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Allow customers to submit return requests to shipments.', 'shipments', 'woocommerce-germanized' ) . '
' . sprintf( _x( 'This option will allow your customers to submit return requests to orders. Return requests will be visible within your %1$s. To learn more about return requests by customers and/or guests, please check the %2$s.', 'shipments', 'woocommerce-germanized' ), '' . _x( 'Return Dashboard', 'shipments', 'woocommerce-germanized' ) . '', '' . _x( 'docs', 'shipments', 'woocommerce-germanized' ) . '' ) . '
', + 'id' => 'supports_customer_returns', + 'placeholder' => '', + 'value' => wc_bool_to_string( $this->get_supports_customer_returns( 'edit' ) ), + 'default' => 'no', + 'type' => 'gzd_toggle', + ), + + array( + 'title' => _x( 'Guest returns', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Allow guests to submit return requests to shipments.', 'shipments', 'woocommerce-germanized' ) . '
' . sprintf( _x( 'Guests will need to provide their email address and the order id to receive a one-time link to submit a return request. The placeholder %s might be used to place the request form on your site.', 'shipments', 'woocommerce-germanized' ), '[gzd_return_request_form]' ) . '
', + 'id' => 'supports_guest_returns', + 'default' => 'no', + 'value' => wc_bool_to_string( $this->get_supports_guest_returns( 'edit' ) ), + 'type' => 'gzd_toggle', + 'custom_attributes' => array( + 'data-show_if_shipping_provider_supports_customer_returns' => '', + ), + ), + + array( + 'title' => _x( 'Manual confirmation', 'shipments', 'woocommerce-germanized' ), + 'desc' => _x( 'Return requests need manual confirmation.', 'shipments', 'woocommerce-germanized' ) . '
' . _x( 'By default return request need manual confirmation e.g. a shop manager needs to review return requests which by default are added with the status "requested" after a customer submitted a return request. If you choose to disable this option, customer return requests will be added as "processing" and an email confirmation including instructions will be sent immediately to the customer.', 'shipments', 'woocommerce-germanized' ) . '
', + 'id' => 'return_manual_confirmation', + 'placeholder' => '', + 'value' => wc_bool_to_string( $this->get_return_manual_confirmation( 'edit' ) ), + 'default' => 'yes', + 'type' => 'gzd_toggle', + 'custom_attributes' => array( + 'data-show_if_shipping_provider_supports_customer_returns' => '', + ), + ), + + array( + 'title' => _x( 'Return instructions', 'shipments', 'woocommerce-germanized' ), + 'desc' => '
' . _x( 'Provide your customer with instructions on how to return the shipment after a return request has been confirmed e.g. explain how to prepare the return for shipment. In case a label cannot be generated automatically, make sure to provide your customer with information on how to obain a return label.', 'shipments', 'woocommerce-germanized' ) . '
', + 'id' => 'return_instructions', + 'placeholder' => '', + 'value' => $this->get_return_instructions( 'edit' ), + 'default' => '', + 'type' => 'textarea', + 'css' => 'width: 100%; min-height: 60px; margin-top: 1em;', + 'custom_attributes' => array( + 'data-show_if_shipping_provider_supports_customer_returns' => '', + ), + ), + ) + ); + + $settings = array_merge( + $settings, + array( + array( + 'type' => 'sectionend', + 'id' => 'shipping_provider_return_options', + ), + ) + ); + + return $settings; + } + + protected function get_all_settings( $for_shipping_method = false ) { + $settings = array(); + $sections = array_keys( $this->get_setting_sections() ); + + foreach ( $sections as $section ) { + $settings[ $section ] = $this->get_settings( $section, $for_shipping_method ); + } + + return $settings; + } + + public function get_shipping_method_settings() { + $settings = $this->get_all_settings( true ); + $sections = $this->get_setting_sections(); + + $method_settings = array(); + $include_current_section = false; + + foreach ( $settings as $section => $section_settings ) { + $global_settings_url = $this->get_edit_link( $section ); + $default_title = $sections[ $section ]; + + foreach ( $section_settings as $setting ) { + $include = false; + $setting = wp_parse_args( + $setting, + array( + 'allow_override' => ( $include_current_section && ! in_array( $setting['type'], array( 'title', 'sectionend' ), true ) ) ? true : false, + 'type' => '', + 'id' => '', + 'value' => '', + 'title_method' => '', + 'title' => '', + 'custom_attributes' => array(), + ) + ); + + if ( true === $setting['allow_override'] ) { + $include = true; + + if ( 'title' === $setting['type'] ) { + $include_current_section = true; + } + } elseif ( $include_current_section && ! in_array( $setting['type'], array( 'title', 'sectionend' ), true ) && false !== $setting['allow_override'] ) { + $include = true; + } elseif ( in_array( $setting['type'], array( 'title', 'sectionend' ), true ) ) { + $include_current_section = false; + } + + if ( $include ) { + $new_setting = array(); + $new_setting['id'] = $this->get_name() . '_' . $setting['id']; + $new_setting['type'] = str_replace( 'gzd_toggle', 'checkbox', $setting['type'] ); + $new_setting['default'] = $setting['value']; + $new_setting['custom_attributes'] = array(); + + if ( ! empty( $setting['custom_attributes'] ) ) { + foreach ( $setting['custom_attributes'] as $attr => $val ) { + $new_attr = $attr; + + if ( 'data-show_if_' === substr( $attr, 0, 13 ) ) { + $new_attr = 'data-show_if_' . $this->get_name() . '_' . substr( $attr, 13, strlen( $attr ) ); + } + + $new_setting['custom_attributes'][ $new_attr ] = $val; + } + } + + if ( 'checkbox' === $new_setting['type'] ) { + $new_setting['label'] = $setting['desc']; + } elseif ( isset( $setting['desc'] ) ) { + $new_setting['description'] = $setting['desc']; + } + + $copy = array( 'options', 'title', 'desc_tip' ); + + foreach ( $copy as $cp ) { + if ( isset( $setting[ $cp ] ) ) { + $new_setting[ $cp ] = $setting[ $cp ]; + } + } + + if ( 'title' === $new_setting['type'] ) { + $new_setting['description'] = sprintf( _x( 'These settings override your global %2$s options. Do only adjust these settings in case you would like to specifically adjust them for this specific shipping method.', 'shipments', 'woocommerce-germanized' ), esc_url( $global_settings_url ), $this->get_title() ); + + if ( empty( $setting['title'] ) ) { + $new_setting['title'] = $default_title; + } + + if ( ! empty( $setting['title_method'] ) ) { + $new_setting['title'] = $setting['title_method']; + } + } + + $method_settings[ $new_setting['id'] ] = $new_setting; + } + } + } + + return $method_settings; + } + + public function get_setting_sections() { + $sections = array( + '' => _x( 'General', 'shipments', 'woocommerce-germanized' ), + ); + + if ( $this->supports_customer_return_requests() ) { + $sections['returns'] = _x( 'Return Requests', 'shipments', 'woocommerce-germanized' ); + } + + return $sections; + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * + * @return ShipmentLabel|false + */ + public function get_label( $shipment ) { + return apply_filters( "{$this->get_hook_prefix()}label", false, $shipment, $this ); + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + */ + public function get_label_fields_html( $shipment ) { + return apply_filters( "{$this->get_hook_prefix()}label_fields_html", '', $shipment, $this ); + } + + /** + * @param \Vendidero\Germanized\Shipments\Shipment $shipment + * @param mixed $props + */ + public function create_label( $shipment, $props = false ) { + $result = new ShipmentError( 'shipping-provider', _x( 'This shipping provider does not support creating labels.', 'shipments', 'woocommerce-germanized' ) ); + + return $result; + } +} diff --git a/packages/woocommerce-germanized-shipments/src/SimpleShipment.php b/packages/woocommerce-germanized-shipments/src/SimpleShipment.php new file mode 100644 index 000000000..3f1a3a983 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/SimpleShipment.php @@ -0,0 +1,379 @@ + 0, + ); + + /** + * Returns the shipment type. + * + * @return string + */ + public function get_type() { + return 'simple'; + } + + /** + * Returns the order id belonging to the shipment. + * + * @param string $context What the value is for. Valid values are 'view' and 'edit'. + * @return integer + */ + public function get_order_id( $context = 'view' ) { + return $this->get_prop( 'order_id', $context ); + } + + /** + * Set shipment order id. + * + * @param string $order_id The order id. + */ + public function set_order_id( $order_id ) { + // Reset order object + $this->order = null; + + $this->set_prop( 'order_id', absint( $order_id ) ); + } + + /** + * Set shipment order. + * + * @param Order $order_shipment The order shipment. + */ + public function set_order_shipment( &$order_shipment ) { + $this->order_shipment = $order_shipment; + } + + /** + * Tries to fetch the order for the current shipment. + * + * @return bool|WC_Order|null + */ + public function get_order() { + if ( is_null( $this->order ) ) { + $this->order = ( $this->get_order_id() > 0 ? wc_get_order( $this->get_order_id() ) : false ); + } + + return $this->order; + } + + /** + * Returns the order shipment instance. Loads from DB if not yet exists. + * + * @return bool|Order + */ + public function get_order_shipment() { + if ( is_null( $this->order_shipment ) ) { + $order = $this->get_order(); + $this->order_shipment = ( $order ? wc_gzd_get_shipment_order( $order ) : false ); + } + + return $this->order_shipment; + } + + /** + * Sync the shipment with it's corresponding order. + * + * @param array $args + * + * @return bool + */ + public function sync( $args = array() ) { + try { + if ( ! $order_shipment = $this->get_order_shipment() ) { + throw new Exception( _x( 'Invalid shipment order', 'shipments', 'woocommerce-germanized' ) ); + } + + /** + * Hotfix WCML infinite loop + * + */ + if ( function_exists( 'wc_gzd_remove_class_filter' ) ) { + wc_gzd_remove_class_filter( 'woocommerce_order_get_items', 'WCML_Orders', 'woocommerce_order_get_items', 10 ); + } + + $order = $order_shipment->get_order(); + + /** + * Make sure that manually adjusted providers are not overridden by syncing. + */ + $default_provider_instance = wc_gzd_get_order_shipping_provider( $order ); + $default_provider = $default_provider_instance ? $default_provider_instance->get_name() : ''; + $provider = $this->get_shipping_provider( 'edit' ); + $address_data = array_merge( + ( $order->has_shipping_address() ? $order->get_address( 'shipping' ) : $order->get_address( 'billing' ) ), + array( + 'email' => $order->get_billing_email(), + 'phone' => $order->get_billing_phone(), + ) + ); + + // Prefer shipping phone in case exists + if ( is_callable( array( $order, 'get_shipping_phone' ) ) && $order->get_shipping_phone() ) { + $address_data['phone'] = $order->get_shipping_phone(); + } + + /** + * Fix to make sure that we are not syncing formatted customer titles (e.g. Herr) + * which prevents shipment addresses from being translated. + */ + if ( isset( $address_data['title'] ) && ! empty( $address_data['title'] ) ) { + if ( $title = $order->get_meta( '_shipping_title', true ) ) { + $address_data['title'] = $title; + } + } + + /** + * Force the country to have a max length of 2. + * https://github.com/woocommerce/woocommerce/issues/27521 + */ + $country = substr( strtoupper( ( $order->has_shipping_address() ? $order->get_shipping_country() : $order->get_billing_country() ) ), 0, 2 ); + $packaging_id = $this->get_packaging_id( 'edit' ); + + $dimensions = array( + 'width' => $this->get_width( 'edit' ), + 'length' => $this->get_length( 'edit' ), + 'height' => $this->get_height( 'edit' ), + ); + + $args = wp_parse_args( + $args, + array( + 'order_id' => $order->get_id(), + 'shipping_method' => wc_gzd_get_shipment_order_shipping_method_id( $order ), + 'shipping_provider' => ( ! empty( $provider ) ) ? $provider : $default_provider, + 'packaging_id' => $this->get_packaging_id( 'edit' ), + 'address' => $address_data, + 'country' => $country, + 'weight' => $this->get_weight( 'edit' ), + 'packaging_weight' => $this->get_packaging_weight( 'edit' ), + 'length' => $dimensions['length'], + 'width' => $dimensions['width'], + 'height' => $dimensions['height'], + 'additional_total' => $order_shipment->calculate_shipment_additional_total( $this ), + ) + ); + + /** + * Filter to allow adjusting the shipment props synced from the corresponding order. + * + * @param mixed $args The properties in key => value pairs. + * @param SimpleShipment $shipment The shipment object. + * @param Order $order_shipment The shipment order object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $args = apply_filters( 'woocommerce_gzd_shipment_sync_props', $args, $this, $order_shipment ); + + $this->set_props( $args ); + + /** + * Action that fires after a shipment has been synced. Syncing is used to + * keep the shipment in sync with the corresponding order. + * + * @param SimpleShipment $shipment The shipment object. + * @param Order $order_shipment The shipment order object. + * @param array $args Array containing properties in key => value pairs to be updated. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_synced', $this, $order_shipment, $args ); + } catch ( Exception $e ) { + return false; + } + + return true; + } + + /** + * Sync items with the corresponding order items. + * Limits quantities and removes non-existing items. + * + * @param array $args + * + * @return bool + */ + public function sync_items( $args = array() ) { + try { + if ( ! $order_shipment = $this->get_order_shipment() ) { + throw new Exception( _x( 'Invalid shipment order', 'shipments', 'woocommerce-germanized' ) ); + } + + $order = $order_shipment->get_order(); + + $args = wp_parse_args( + $args, + array( + 'items' => array(), + ) + ); + + $available_items = $order_shipment->get_available_items_for_shipment( + array( + 'shipment_id' => $this->get_id(), + 'exclude_current_shipment' => true, + ) + ); + + foreach ( $available_items as $item_id => $item_data ) { + if ( $order_item = $order->get_item( $item_id ) ) { + $quantity = $item_data['max_quantity']; + + if ( ! empty( $args['items'] ) ) { + if ( isset( $args['items'][ $item_id ] ) ) { + $new_quantity = absint( $args['items'][ $item_id ] ); + + if ( $new_quantity < $quantity ) { + $quantity = $new_quantity; + } + } else { + continue; + } + } + + if ( ! $shipment_item = $this->get_item_by_order_item_id( $item_id ) ) { + $shipment_item = wc_gzd_create_shipment_item( $this, $order_item, array( 'quantity' => $quantity ) ); + + $this->add_item( $shipment_item ); + } else { + $shipment_item->sync( array( 'quantity' => $quantity ) ); + } + } + } + + foreach ( $this->get_items() as $item ) { + // Remove non-existent items + if ( ! $order_item = $order->get_item( $item->get_order_item_id() ) ) { + $this->remove_item( $item->get_id() ); + } + } + + // Sync packaging + $this->sync_packaging(); + + /** + * Action that fires after items of a shipment have been synced. + * + * @param SimpleShipment $shipment The shipment object. + * @param Order $order_shipment The shipment order object. + * @param array $args Array containing additional data e.g. items. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_items_synced', $this, $order_shipment, $args ); + + } catch ( Exception $e ) { + return false; + } + + return true; + } + + /** + * Returns available shipment methods by checking the corresponding order. + * + * @return string[] + */ + public function get_available_shipping_methods() { + $methods = array(); + + if ( $order = $this->get_order() ) { + $items = $order->get_shipping_methods(); + + foreach ( $items as $item ) { + $methods[ $item->get_method_id() . ':' . $item->get_instance_id() ] = $item->get_name(); + } + } + + return $methods; + } + + /** + * Returns the number of items available for shipment. + * + * @return int|mixed|void + */ + public function get_shippable_item_count() { + if ( $order_shipment = $this->get_order_shipment() ) { + return $order_shipment->get_shippable_item_count(); + } + + return 0; + } + + /** + * Returns whether the Shipment needs additional items or not. + * + * @param bool|integer[] $available_items + * + * @return bool + */ + public function needs_items( $available_items = false ) { + + if ( ! $available_items && ( $order = wc_gzd_get_shipment_order( $this->get_order() ) ) ) { + $available_items = array_keys( $order->get_available_items_for_shipment() ); + } + + return ( $this->is_editable() && ! $this->contains_order_item( $available_items ) ); + } + + /** + * Returns the edit shipment URL. + * + * @return mixed|string|void + */ + public function get_edit_shipment_url() { + /** + * Filter to adjust the edit Shipment admin URL. + * + * The dynamic portion of this hook, `$this->get_hook_prefix()` is used to construct a + * unique hook for a shipment type. + * + * Example hook name: woocommerce_gzd_shipment_get_edit_url + * + * @param string $url The URL. + * @param Shipment $this The shipment object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + return apply_filters( "{$this->get_hook_prefix()}edit_url", get_admin_url( null, 'post.php?post=' . $this->get_order_id() . '&action=edit&shipment_id=' . $this->get_id() ), $this ); + } +} diff --git a/packages/woocommerce-germanized-shipments/src/Validation.php b/packages/woocommerce-germanized-shipments/src/Validation.php new file mode 100644 index 000000000..c69a218aa --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/Validation.php @@ -0,0 +1,234 @@ +get_id() ) { + self::new_order( $order ); + } + }, + 300, + 1 + ); + }, + 10, + 1 + ); + + add_action( 'woocommerce_delete_order', array( __CLASS__, 'delete_order' ), 10, 1 ); + + foreach ( array( 'cancelled', 'failed', 'refunded' ) as $cancelled_status ) { + add_action( "woocommerce_order_status_{$cancelled_status}", array( __CLASS__, 'maybe_cancel_shipments' ), 10, 2 ); + } + + add_action( 'before_delete_post', array( __CLASS__, 'before_delete_refund' ), 10, 1 ); + add_action( 'woocommerce_delete_order_refund', array( __CLASS__, 'delete_refund_order' ), 10, 1 ); + add_action( 'woocommerce_order_refund_object_updated_props', array( __CLASS__, 'refresh_refund_order' ), 10, 1 ); + + // Check if order is shipped + add_action( 'woocommerce_gzd_shipment_status_changed', array( __CLASS__, 'maybe_update_order_date_shipped' ), 10, 4 ); + + add_action( 'woocommerce_gzd_shipping_provider_deactivated', array( __CLASS__, 'maybe_disable_default_shipping_provider' ), 10 ); + } + + /** + * In case a certain shipping provider is being deactivated make sure that the default + * shipping provider option is removed in case the option equals the deactivated provider. + * + * @param ShippingProvider $provider + */ + public static function maybe_disable_default_shipping_provider( $provider ) { + $default_provider = wc_gzd_get_default_shipping_provider(); + + if ( $default_provider === $provider->get_name() ) { + update_option( 'woocommerce_gzd_shipments_default_shipping_provider', '' ); + } + } + + /** + * @param $shipment_id + * @param $status_from + * @param $status_to + * @param Shipment $shipment + */ + public static function maybe_update_order_date_shipped( $shipment_id, $status_from, $status_to, $shipment ) { + if ( 'simple' === $shipment->get_type() && ( $order = $shipment->get_order() ) ) { + self::check_order_shipped( $order ); + } + } + + public static function check_order_shipped( $order ) { + if ( $shipment_order = wc_gzd_get_shipment_order( $order ) ) { + if ( $shipment_order->is_shipped() ) { + /** + * Action that fires as soon as an order has been shipped completely. + * That is the case when the order contains all relevant shipments and all the shipments are marked as shipped. + * + * @param string $order_id The order id. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipments_order_shipped', $shipment_order->get_order()->get_id() ); + + $shipment_order->get_order()->update_meta_data( '_date_shipped', time() ); + $shipment_order->get_order()->save(); + } else { + $shipment_order->get_order()->delete_meta_data( '_date_shipped' ); + $shipment_order->get_order()->save(); + } + } + } + + /** + * Delete editable shipments if an order is cancelled. + * + * @param $order_id + * @param WC_Order $order + */ + public static function maybe_cancel_shipments( $order_id, $order ) { + $shipments = wc_gzd_get_shipments_by_order( $order ); + + foreach ( $shipments as $shipment ) { + if ( $shipment->is_editable() ) { + $shipment->delete(); + } + } + } + + public static function before_delete_refund( $refund_id ) { + if ( $refund = wc_get_order( $refund_id ) ) { + + if ( is_a( $refund, 'WC_Order_Refund' ) ) { + self::$current_refund_parent_order = $refund->get_parent_id(); + } + } + } + + public static function delete_refund_order( $refund_id ) { + if ( false !== self::$current_refund_parent_order ) { + + if ( $order_shipment = wc_gzd_get_shipment_order( self::$current_refund_parent_order ) ) { + $order_shipment->validate_shipments(); + } + + self::$current_refund_parent_order = false; + } + } + + public static function refresh_refund_order( $refund ) { + if ( $refund->get_parent_id() <= 0 ) { + return; + } + + if ( $order_shipment = wc_gzd_get_shipment_order( $refund->get_parent_id() ) ) { + $order_shipment->validate_shipments(); + } + } + + public static function delete_order( $order_id ) { + if ( $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) { + + foreach ( $order_shipment->get_shipments() as $shipment ) { + + if ( $shipment->is_editable() ) { + $order_shipment->remove_shipment( $shipment->get_id() ); + } + } + + $order_shipment->save(); + } + } + + public static function new_order( $order ) { + if ( $order_shipment = wc_gzd_get_shipment_order( $order ) ) { + $order_shipment->validate_shipments(); + } + } + + public static function update_order( $order_id ) { + if ( $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) { + $order_shipment->validate_shipments(); + } + } + + public static function delete_order_item( $order_item_id ) { + try { + if ( $order_id = wc_get_order_id_by_order_item_id( $order_item_id ) ) { + + if ( $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) { + foreach ( $order_shipment->get_shipments() as $shipment ) { + + if ( $shipment->is_editable() ) { + if ( $item = $shipment->get_item_by_order_item_id( $order_item_id ) ) { + $shipment->remove_item( $item->get_id() ); + } + } + } + + $order_shipment->save(); + } + } + } catch ( Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch + } + } + + public static function create_order_item( $order_item_id, $order_item, $order_id ) { + if ( $order_shipment = wc_gzd_get_shipment_order( $order_id ) ) { + $order_shipment->validate_shipments(); + } + } + + protected static function is_admin_save_order_request() { + $is_admin_order_save_request = doing_action( 'save_post' ); + + /** + * Detect admin order adjustments e.g. add item, remove item, save post etc. and + * prevent singular order item hooks from executing to prevent multiple shipment validation requests + * which will execute on order save hook as well. + */ + if ( ! $is_admin_order_save_request && wp_doing_ajax() && isset( $_REQUEST['action'] ) && isset( $_REQUEST['order_id'] ) && strpos( wc_clean( wp_unslash( $_REQUEST['action'] ) ), 'woocommerce_' ) !== false ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $is_admin_order_save_request = true; + } + + return $is_admin_order_save_request; + } + + /** + * @param $order_item_id + * @param WC_Order_Item $order_item + */ + public static function update_order_item( $order_item_id, $order_item ) { + if ( ! self::is_admin_save_order_request() ) { + if ( is_callable( array( $order_item, 'get_order_id' ) ) ) { + + if ( $order_shipment = wc_gzd_get_shipment_order( $order_item->get_order_id() ) ) { + $order_shipment->validate_shipments(); + } + } + } + } +} diff --git a/packages/woocommerce-germanized-shipments/src/WPMLHelper.php b/packages/woocommerce-germanized-shipments/src/WPMLHelper.php new file mode 100644 index 000000000..f19e28653 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/src/WPMLHelper.php @@ -0,0 +1,166 @@ +get_order_id() ) ) { + $language = $order->get_meta( 'wpml_language', true ); + + foreach ( $args['items'] as $key => $item ) { + $id = $item->get_product_id(); + $id = apply_filters( 'wpml_object_id', $id, get_post_type( $id ), true, $language ); + + if ( $product = wc_get_product( $id ) ) { + $args['items'][ $key ]->set_name( $product->get_name() ); + } + } + } + + return $args; + } + + public static function register_provider_filters() { + add_filter( 'woocommerce_gzd_shipping_provider_get_tracking_desc_placeholder', array( __CLASS__, 'filter_shipping_provider_placeholder' ), 10, 2 ); + add_filter( 'woocommerce_gzd_shipping_provider_get_tracking_url_placeholder', array( __CLASS__, 'filter_shipping_provider_url' ), 10, 2 ); + add_filter( 'woocommerce_gzd_shipping_provider_get_return_instructions', array( __CLASS__, 'filter_shipping_provider_return_instructions' ), 10, 2 ); + + foreach ( Helper::instance()->get_shipping_providers() as $provider ) { + add_filter( "woocommerce_gzd_shipping_provider_{$provider->get_name()}_get_tracking_desc_placeholder", array( __CLASS__, 'filter_shipping_provider_placeholder' ), 10, 2 ); + add_filter( "woocommerce_gzd_shipping_provider_{$provider->get_name()}_get_tracking_url_placeholder", array( __CLASS__, 'filter_shipping_provider_url' ), 10, 2 ); + add_filter( "woocommerce_gzd_shipping_provider_{$provider->get_name()}_get_return_instructions", array( __CLASS__, 'filter_shipping_provider_return_instructions' ), 10, 2 ); + } + } + + public static function filter_shipping_provider_return_instructions( $instructions, $provider ) { + $string_name = 'return_instructions'; + $translated_string = apply_filters( 'wpml_translate_string', $instructions, self::get_shipping_provider_string_id( $string_name, $provider ), self::get_shipping_provider_string_package( $string_name, $provider ) ); + + return $translated_string; + } + + public static function filter_shipping_provider_url( $placeholder, $provider ) { + $string_name = 'tracking_url_placeholder'; + $translated_string = apply_filters( 'wpml_translate_string', $placeholder, self::get_shipping_provider_string_id( $string_name, $provider ), self::get_shipping_provider_string_package( $string_name, $provider ) ); + + return $translated_string; + } + + public static function filter_shipping_provider_placeholder( $placeholder, $provider ) { + $string_name = 'tracking_desc_placeholder'; + $translated_string = apply_filters( 'wpml_translate_string', $placeholder, self::get_shipping_provider_string_id( $string_name, $provider ), self::get_shipping_provider_string_package( $string_name, $provider ) ); + + return $translated_string; + } + + /** + * @param integer $provider_id + * @param Simple $provider + */ + public static function register_shipping_provider_strings( $provider_id, $provider ) { + + foreach ( self::get_shipping_provider_strings() as $string_name => $title ) { + $title = sprintf( $title, $provider->get_title() ); + $getter = "get_{$string_name}"; + + if ( is_callable( array( $provider, $getter ) ) ) { + $value = $provider->{$getter}(); + + do_action( 'wpml_register_string', $value, self::get_shipping_provider_string_id( $string_name, $provider ), self::get_shipping_provider_string_package( $string_name, $provider ), $title, 'AREA' ); + } + } + } + + protected static function get_shipping_provider_strings() { + $strings = array( + 'tracking_desc_placeholder' => _x( '%s tracking description', 'shipments', 'woocommerce-germanized' ), + 'tracking_url_placeholder' => _x( '%s tracking URL', 'shipments', 'woocommerce-germanized' ), + 'return_instructions' => _x( '%s return instructions', 'shipments', 'woocommerce-germanized' ), + ); + + return $strings; + } + + /** + * @param $string_name + * @param Simple $provider + */ + protected static function get_shipping_provider_string_id( $string_name, $provider ) { + return "woocommerce_gzd_shipping_provider_{$provider->get_name()}_{$string_name}"; + } + + /** + * @param $string_name + * @param Simple $provider + */ + protected static function get_shipping_provider_string_package( $string_name, $provider ) { + $strings = self::get_shipping_provider_strings(); + $package = array(); + + if ( array_key_exists( $string_name, $strings ) ) { + $title = sprintf( $strings[ $string_name ], $provider->get_title() ); + + $package = array( + 'kind' => 'Shipping Provider', + 'name' => "{$provider->get_name()}_{$string_name}", + 'edit_link' => $provider->get_edit_link(), + 'title' => $title, + ); + } + + return $package; + } + + /** + * @param $emails + */ + public static function register_emails( $emails ) { + $emails['WC_GZD_Email_Customer_Shipment'] = 'customer_shipment'; + $emails['WC_GZD_Email_Customer_Return_Shipment'] = 'customer_return_shipment'; + $emails['WC_GZD_Email_Customer_Return_Shipment_Delivered'] = 'customer_return_shipment_delivered'; + $emails['WC_GZD_Email_Customer_Guest_Return_Shipment_Request'] = 'customer_guest_return_shipment_request'; + + return $emails; + } +} diff --git a/packages/woocommerce-germanized-shipments/templates/emails/admin-new-return-shipment-request.php b/packages/woocommerce-germanized-shipments/templates/emails/admin-new-return-shipment-request.php new file mode 100644 index 000000000..0b3d10afb --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/admin-new-return-shipment-request.php @@ -0,0 +1,60 @@ + + + +

get_formatted_sender_full_name() ) ); ?>

+ + + +

get_billing_first_name() ) ); // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch ?>

+ +

+ get_order_number() ) ); ?> +

+ +

+ +

+ +

+ + + +

get_billing_first_name() ) ); // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch ?>

+ +

+ +

+ + + +

get_billing_first_name() ) ); // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch ?>

+ +

+ +

+ + + +

get_billing_first_name() ) ); // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch ?>

+ +

+ +

+ + + + + +
+ $shipment ) : + $count++; + ?> + 1 ) : ?> +

+ + + get_est_delivery_date() ) : ?> +

get_est_delivery_date(), wc_date_format() ) ); ?>

+ + + has_tracking() ) : ?> + get_tracking_url() ) : ?> +

+ + + has_tracking_instruction() ) : ?> +

get_tracking_instruction() ); ?>

+ + +

+ + +
diff --git a/packages/woocommerce-germanized-shipments/templates/emails/email-return-shipment-instructions.php b/packages/woocommerce-germanized-shipments/templates/emails/email-return-shipment-instructions.php new file mode 100644 index 000000000..2789a1d36 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/email-return-shipment-instructions.php @@ -0,0 +1,26 @@ +get_shipping_provider_instance(); +?> + +has_return_instructions() ) : ?> +

get_return_instructions() ) ) . PHP_EOL ); ?>

+ diff --git a/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-address.php b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-address.php new file mode 100644 index 000000000..f355c2f80 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-address.php @@ -0,0 +1,34 @@ + + + + + + +
+

+ +
+ get_formatted_address() ); ?> +
+
diff --git a/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-details.php b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-details.php new file mode 100644 index 000000000..a6ed2569d --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-details.php @@ -0,0 +1,87 @@ + + +

+ get_edit_shipment_url() ) . '">'; + $after = ''; + } else { + $before = ''; + $after = ''; + } + /* translators: %s: Order ID. */ + echo wp_kses_post( $before . ( ! $sent_to_admin ? sprintf( _x( 'Details to your %s', 'shipments', 'woocommerce-germanized' ), wc_gzd_get_shipment_label_title( $shipment->get_type() ) ) : sprintf( _x( '[%1$s #%2$s]', 'shipments', 'woocommerce-germanized' ), wc_gzd_get_shipment_label_title( $shipment->get_type() ), $shipment->get_shipment_number() ) ) . $after ); + ?> +

+ +
+ + + + + + + + + $sent_to_admin, + 'show_image' => false, + 'image_size' => array( 32, 32 ), + 'plain_text' => $plain_text, + 'sent_to_admin' => $sent_to_admin, + ) + ); + ?> + +
+
+ + diff --git a/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-items.php b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-items.php new file mode 100644 index 000000000..e17d2f062 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-items.php @@ -0,0 +1,114 @@ + $item ) : + $product = $item->get_product(); + $sku = $item->get_sku(); + $purchase_note = ''; + $image = ''; + + /** + * Filter to decide whether a specific ShipmentItem is visible within email table or not. + * + * @param boolean $is_visible Whether the ShipmentItem is visible or not. + * @param ShipmentItem $item The ShipmentItem object. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + if ( ! apply_filters( 'woocommerce_gzd_shipment_item_visible', true, $item ) ) { + continue; + } + + if ( is_object( $product ) ) { + $image = $product->get_image( $image_size ); + } + + ?> + + + get_name(), $item, false ) ); + + // SKU. + if ( $show_sku && $sku ) { + echo wp_kses_post( ' (#' . $sku . ')' ); + } + + /* + * Action that fires while outputting meta data for a ShipmentItem table display in an Email. + * + * @param integer $item_id The shipment item id. + * @param \Vendidero\Germanized\Shipments\ShipmentItem $item The shipment item instance. + * @param \Vendidero\Germanized\Shipments\Shipment $shipment The shipment instance. + * @param boolean $plain_text Whether this email is in plaintext format or not. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_item_meta', $item_id, $item, $shipment, $plain_text ); + + ?> + + + get_quantity(), $item ) ); + ?> + + + + diff --git a/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-tracking.php b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-tracking.php new file mode 100644 index 000000000..4d5cf3d4a --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/email-shipment-tracking.php @@ -0,0 +1,42 @@ + + + + + + +
+

+ + get_est_delivery_date() ) : ?> +

get_est_delivery_date(), wc_date_format() ) ); ?>

+ + + get_tracking_url() ) : ?> +

+ + + has_tracking_instruction() ) : ?> +

get_tracking_instruction() ); ?>

+ +
diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/admin-new-return-shipment-request.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/admin-new-return-shipment-request.php new file mode 100644 index 000000000..14481e946 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/admin-new-return-shipment-request.php @@ -0,0 +1,41 @@ +get_formatted_sender_full_name() ) ) . "\n\n"; + +echo "\n\n"; + +/* This hook is documented in templates/emails/customer-shipment.php */ +do_action( 'woocommerce_gzd_email_shipment_details', $shipment, $sent_to_admin, $plain_text, $email ); + +/** + * Show user-defined additional content - this is set in each email's settings. + */ +if ( $additional_content ) { + echo esc_html( wp_strip_all_tags( wptexturize( $additional_content ) ) ); + echo "\n\n----------------------------------------\n\n"; +} + +echo "\n----------------------------------------\n\n"; + +echo wp_kses_post( apply_filters( 'woocommerce_email_footer_text', get_option( 'woocommerce_email_footer_text' ) ) ); diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-guest-return-shipment-request.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-guest-return-shipment-request.php new file mode 100644 index 000000000..132b76d58 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-guest-return-shipment-request.php @@ -0,0 +1,40 @@ +get_billing_first_name() ) ) . "\n\n"; // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + +echo sprintf( esc_html_x( 'You\'ve requested a return to your order %s. Please follow the link to add your return request.', 'shipments', 'woocommerce-germanized' ), esc_html( $order->get_order_number() ) ) . "\n\n"; + +echo esc_url( $add_return_request_url ) . "\n\n"; + +/** + * Show user-defined additional content - this is set in each email's settings. + */ +if ( $additional_content ) { + echo esc_html( wp_strip_all_tags( wptexturize( $additional_content ) ) ); + echo "\n\n----------------------------------------\n\n"; +} + +echo "\n----------------------------------------\n\n"; + +echo wp_kses_post( apply_filters( 'woocommerce_email_footer_text', get_option( 'woocommerce_email_footer_text' ) ) ); diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-return-shipment-delivered.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-return-shipment-delivered.php new file mode 100644 index 000000000..4c5660934 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-return-shipment-delivered.php @@ -0,0 +1,43 @@ +get_billing_first_name() ) ) . "\n\n"; // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + +echo esc_html_x( 'Thank you! Your return has been received successfully. There are more details below for your reference:', 'shipments', 'woocommerce-germanized' ); + +echo "\n\n"; + +/* This hook is documented in templates/emails/customer-shipment.php */ +do_action( 'woocommerce_gzd_email_shipment_details', $shipment, $sent_to_admin, $plain_text, $email ); + +/** + * Show user-defined additional content - this is set in each email's settings. + */ +if ( $additional_content ) { + echo esc_html( wp_strip_all_tags( wptexturize( $additional_content ) ) ); + echo "\n\n----------------------------------------\n\n"; +} + +echo "\n----------------------------------------\n\n"; + +echo wp_kses_post( apply_filters( 'woocommerce_email_footer_text', get_option( 'woocommerce_email_footer_text' ) ) ); diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-return-shipment.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-return-shipment.php new file mode 100644 index 000000000..6f2fcdcd7 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-return-shipment.php @@ -0,0 +1,48 @@ +get_billing_first_name() ) ) . "\n\n"; // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + + +if ( $is_confirmation ) { + echo esc_html_x( 'Your return request has been accepted. Please follow the instructions beneath to return your shipment.', 'shipments', 'woocommerce-germanized' ); +} else { + echo esc_html_x( 'A new return has been added to your order. Please follow the instructions beneath to return your shipment.', 'shipments', 'woocommerce-germanized' ); +} + +echo "\n\n"; + +/* This hook is documented in templates/emails/customer-shipment.php */ +do_action( 'woocommerce_gzd_email_shipment_details', $shipment, $sent_to_admin, $plain_text, $email ); + +/** + * Show user-defined additional content - this is set in each email's settings. + */ +if ( $additional_content ) { + echo esc_html( wp_strip_all_tags( wptexturize( $additional_content ) ) ); + echo "\n\n----------------------------------------\n\n"; +} + +echo "\n----------------------------------------\n\n"; + +echo wp_kses_post( apply_filters( 'woocommerce_email_footer_text', get_option( 'woocommerce_email_footer_text' ) ) ); diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-shipment.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-shipment.php new file mode 100644 index 000000000..751b764c4 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/customer-shipment.php @@ -0,0 +1,49 @@ +get_billing_first_name() ) ) . "\n\n"; // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch + +if ( $partial_shipment ) { + /* translators: %s: Site title */ + printf( esc_html_x( 'Your order on %1$s has been partially shipped via %2$s. Find details below for your reference:', 'shipments', 'woocommerce-germanized' ), wp_specialchars_decode( get_option( 'blogname' ), ENT_QUOTES ), wc_gzd_get_shipment_shipping_provider_title( $shipment ) ); // phpcs:ignore WordPress.XSS.EscapeOutput.OutputNotEscaped +} else { + /* translators: %s: Site title */ + printf( esc_html_x( 'Your order on %1$s has been shipped via %2$s. Find details below for your reference:', 'shipments', 'woocommerce-germanized' ), wp_specialchars_decode( get_option( 'blogname' ), ENT_QUOTES ), wc_gzd_get_shipment_shipping_provider_title( $shipment ) ); // phpcs:ignore WordPress.XSS.EscapeOutput.OutputNotEscaped +} + +echo "\n\n"; + +/* This hook is documented in templates/emails/customer-shipment.php */ +do_action( 'woocommerce_gzd_email_shipment_details', $shipment, $sent_to_admin, $plain_text, $email ); + +/** + * Show user-defined additional content - this is set in each email's settings. + */ +if ( $additional_content ) { + echo esc_html( wp_strip_all_tags( wptexturize( $additional_content ) ) ); + echo "\n\n----------------------------------------\n\n"; +} + +echo "\n----------------------------------------\n\n"; + +echo wp_kses_post( apply_filters( 'woocommerce_email_footer_text', get_option( 'woocommerce_email_footer_text' ) ) ); diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/email-order-shipments.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-order-shipments.php new file mode 100644 index 000000000..2c8b85934 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-order-shipments.php @@ -0,0 +1,47 @@ + $shipment ) { + $count++; + + echo "\n"; + + if ( count( $shipments ) > 1 ) { + echo sprintf( esc_html_x( 'Shipment %1$d of %2$d', 'shipments', 'woocommerce-germanized' ), esc_html( $count ), esc_html( count( $shipments ) ) ) . "\n\n"; + } + + if ( $shipment->get_est_delivery_date() ) { + echo esc_html( _x( 'Estimated date:', 'shipments', 'woocommerce-germanized' ) ) . ' ' . esc_html( wc_format_datetime( $shipment->get_est_delivery_date(), wc_date_format() ) ) . "\n\n"; + } + + if ( $shipment->has_tracking() ) { + if ( $shipment->get_tracking_url() ) { + echo esc_html( _x( 'Track your shipment', 'shipments', 'woocommerce-germanized' ) ) . ': ' . esc_url( $shipment->get_tracking_url() ) . "\n"; + } + + if ( $shipment->has_tracking_instruction() ) { + echo esc_html( $shipment->get_tracking_instruction( true ) ) . "\n"; + } + } else { + echo esc_html( _x( 'Sorry, this shipment does currently not support tracking.', 'shipments', 'woocommerce-germanized' ) ) . "\n"; + } +} diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/email-return-shipment-instructions.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-return-shipment-instructions.php new file mode 100644 index 000000000..ee24b7d3b --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-return-shipment-instructions.php @@ -0,0 +1,23 @@ +get_shipping_provider_instance(); + +if ( $provider && $provider->has_return_instructions() ) { + echo wp_kses_post( wpautop( wptexturize( $provider->get_return_instructions() ) ) . PHP_EOL ); +} diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-address.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-address.php new file mode 100644 index 000000000..2eaac15ae --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-address.php @@ -0,0 +1,20 @@ +#i', "\n", $shipment->get_formatted_address() ) . "\n"; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-details.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-details.php new file mode 100644 index 000000000..51c4b8206 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-details.php @@ -0,0 +1,43 @@ +get_type() ) ) : sprintf( _x( '[%1$s #%2$s]', 'shipments', 'woocommerce-germanized' ), wc_gzd_get_shipment_label_title( $shipment->get_type() ), $shipment->get_shipment_number() ) ) ) ) . "\n"; +echo "\n" . wc_gzd_get_email_shipment_items( // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + $shipment, + array( + 'show_sku' => $sent_to_admin, + 'show_image' => false, + 'image_size' => array( 32, 32 ), + 'plain_text' => true, + 'sent_to_admin' => $sent_to_admin, + ) +); + +echo "==========\n\n"; + +if ( $sent_to_admin ) { + /* translators: %s: Shipment link. */ + echo "\n" . sprintf( esc_html_x( 'View shipment: %s', 'shipments', 'woocommerce-germanized' ), esc_url( $shipment->get_edit_shipment_url() ) ) . "\n"; +} + +/* This hook is documented in templates/emails/customer-shipment.php */ +do_action( 'woocommerce_gzd_email_after_shipment_table', $shipment, $sent_to_admin, $plain_text, $email ); diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-items.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-items.php new file mode 100644 index 000000000..2fa1d8308 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-items.php @@ -0,0 +1,46 @@ + $item ) : + $product = $item->get_product(); + $sku = $item->get_sku(); + $purchase_note = ''; + + /* This filter is documented in templates/emails/email-shipment-items.php */ + if ( ! apply_filters( 'woocommerce_gzd_shipment_item_visible', true, $item ) ) { + continue; + } + + /* This filter is documented in templates/emails/email-shipment-items.php */ + echo wp_kses_post( apply_filters( 'woocommerce_gzd_shipment_item_name', $item->get_name(), $item, false ) ); + + if ( $show_sku && $sku ) { + echo ' (#' . esc_html( $sku ) . ')'; + } + + /* This filter is documented in templates/emails/email-shipment-items.php */ + echo ' X ' . wp_kses_post( apply_filters( 'woocommerce_gzd_email_shipment_item_quantity', $item->get_quantity(), $item ) ); + echo "\n"; + + /* This hook is documented in templates/emails/email-shipment-items.php */ + do_action( 'woocommerce_gzd_shipment_item_meta', $item_id, $item, $shipment, $plain_text ); + + echo "\n\n"; +endforeach; diff --git a/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-tracking.php b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-tracking.php new file mode 100644 index 000000000..403a94c5a --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/emails/plain/email-shipment-tracking.php @@ -0,0 +1,33 @@ +get_est_delivery_date() ) { + echo esc_html( _x( 'Estimated date:', 'shipments', 'woocommerce-germanized' ) ) . ' ' . esc_html( wc_format_datetime( $shipment->get_est_delivery_date(), wc_date_format() ) ) . "\n\n"; +} + +if ( $shipment->get_tracking_url() ) { + echo esc_html( _x( 'Track your shipment', 'shipments', 'woocommerce-germanized' ) ) . ': ' . esc_url( $shipment->get_tracking_url() ) . "\n"; +} + +if ( $shipment->has_tracking_instruction() ) { + echo esc_html( $shipment->get_tracking_instruction( true ) ) . "\n"; +} diff --git a/packages/woocommerce-germanized-shipments/templates/global/empty.php b/packages/woocommerce-germanized-shipments/templates/global/empty.php new file mode 100644 index 000000000..fd3489d94 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/global/empty.php @@ -0,0 +1,21 @@ + +
> + + + + + +

+ + +

+ +

+ + +

+ +
+ + + +

+ + +

+ +
+ + + +
diff --git a/packages/woocommerce-germanized-shipments/templates/myaccount/add-return-shipment.php b/packages/woocommerce-germanized-shipments/templates/myaccount/add-return-shipment.php new file mode 100644 index 000000000..9838f1e98 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/myaccount/add-return-shipment.php @@ -0,0 +1,96 @@ + +

+ +

+ + +

+ +

+ + +
+ + + + + + + + + + + + get_available_items_for_return() as $order_item_id => $item_data ) { + + wc_get_template( + 'shipment/add-return-shipment-item.php', + array( + 'item' => $shipment_order->get_simple_shipment_item( $order_item_id ), + 'order_item_id' => $order_item_id, + 'order' => $order, + 'max_quantity' => $item_data['max_quantity'], + ) + ); + } + + /** + * This action is executed after printing the add return shipment table on the customer account page. + * + * @param WC_Order $order The order instance. + * + * @since 3.1.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_add_return_shipment_details_after_shipment_table_items', $order ); + ?> + +
+ +

+ + + + + + +

+
+ diff --git a/packages/woocommerce-germanized-shipments/templates/myaccount/order-shipments.php b/packages/woocommerce-germanized-shipments/templates/myaccount/order-shipments.php new file mode 100644 index 000000000..ed4571e3e --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/myaccount/order-shipments.php @@ -0,0 +1,53 @@ + + + +

+ + 'simple', + 'shipments' => $shipments, + 'order' => $order, + ) + ); + ?> + + + +

+ +

+ + + +

+ + 'return', + 'shipments' => $returns, + 'order' => $order, + ) + ); + ?> + diff --git a/packages/woocommerce-germanized-shipments/templates/myaccount/shipments.php b/packages/woocommerce-germanized-shipments/templates/myaccount/shipments.php new file mode 100644 index 000000000..71d4cf1fc --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/myaccount/shipments.php @@ -0,0 +1,113 @@ + + + + + + $column_name ) : ?> + + + + + + + get_item_count(); + ?> + + get_type() ) as $column_id => $column_name ) : ?> + + + + + + + + diff --git a/packages/woocommerce-germanized-shipments/templates/myaccount/view-shipment.php b/packages/woocommerce-germanized-shipments/templates/myaccount/view-shipment.php new file mode 100644 index 000000000..7c28e5636 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/myaccount/view-shipment.php @@ -0,0 +1,42 @@ + +

+ ' . $shipment->get_shipment_number() . '', // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + '' . wc_format_datetime( $shipment->get_date_created() ) . '', // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + '' . wc_gzd_get_shipment_status_name( $shipment->get_status() ) . '' // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + ); + ?> +

+ + diff --git a/packages/woocommerce-germanized-shipments/templates/shipment/add-return-shipment-item.php b/packages/woocommerce-germanized-shipments/templates/shipment/add-return-shipment-item.php new file mode 100644 index 000000000..d4339fa15 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/shipment/add-return-shipment-item.php @@ -0,0 +1,90 @@ + + + + + + + + + get_product(); + $is_visible = $product && $product->is_visible(); + $item_sku = $item->get_sku(); + + /** This filter is documented in templates/myaccount/shipment/shipment-details-item.php */ + $product_permalink = apply_filters( 'woocommerce_gzd_shipment_item_permalink', $is_visible ? $product->get_permalink() : '', $item, $order ); + + /** This filter is documented in templates/emails/email-shipment-items.php */ + echo apply_filters( 'woocommerce_gzd_shipment_item_name', ( $product_permalink ? sprintf( '%s', esc_url( $product_permalink ), $item->get_name() ) : $item->get_name() ) . ( ! empty( $item_sku ) ? ' (' . esc_html( $item_sku ) . ')' : '' ), $item, $is_visible ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + ?> + + + + + + + + + 1 + + 'item[' . esc_attr( $order_item_id ) . '][quantity]', + 'input_value' => 1, + 'max_value' => $max_quantity, + 'min_value' => 1, + ), + $item->get_product() + ); + ?> + + diff --git a/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-address.php b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-address.php new file mode 100644 index 000000000..6e0b8713c --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-address.php @@ -0,0 +1,52 @@ + +
+ +
+ +

+ +
+ get_formatted_address( esc_html_x( 'N/A', 'shipments', 'woocommerce-germanized' ) ) ); ?> + + get_phone() ) : ?> +

get_phone() ); ?>

+ + + get_email() ) : ?> + + +
+ +
+ +
+ + diff --git a/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-item.php b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-item.php new file mode 100644 index 000000000..cee0cfd43 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-item.php @@ -0,0 +1,93 @@ + + + + + is_visible(); + $item_sku = $item->get_sku(); + + /** + * This filter may adjust the shipment item permalink on the customer account page. + * + * @param string $permalink The permalink. + * @param ShipmentItem $item The shipment item instance. + * @param Shipment $shipment The shipment instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + $product_permalink = apply_filters( 'woocommerce_gzd_shipment_item_permalink', $is_visible ? $product->get_permalink() : '', $item, $shipment ); + + /** This filter is documented in templates/emails/email-shipment-items.php */ + echo apply_filters( 'woocommerce_gzd_shipment_item_name', ( $product_permalink ? sprintf( '%s', esc_url( $product_permalink ), $item->get_name() ) : $item->get_name() ) . ( ! empty( $item_sku ) ? ' (' . esc_html( $item_sku ) . ')' : '' ), $item, $is_visible ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + ?> + + + + get_quantity(); + $qty_display = esc_html( $qty ); + + /** + * This filter may adjust the shipment item quantity HTML on the customer account page. + * + * @param string $html The HTML output. + * @param ShipmentItem $item The shipment item instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + echo apply_filters( 'woocommerce_gzd_shipment_item_quantity_html', ' ' . sprintf( '× %s', $qty_display ) . '', $item ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped + ?> + + + diff --git a/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-tracking.php b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-tracking.php new file mode 100644 index 000000000..6cb603862 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details-tracking.php @@ -0,0 +1,44 @@ + +
+ +

+ + get_tracking_url() ) : ?> +

+ + + has_tracking_instruction() ) : ?> +

get_tracking_instruction() ); ?>

+ + +
+ + diff --git a/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details.php b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details.php new file mode 100644 index 000000000..0050f657e --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-details.php @@ -0,0 +1,123 @@ +get_order(); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited +$show_receiver_details = is_user_logged_in() && $order && $order->get_user_id() === get_current_user_id(); +$show_tracking = $show_receiver_details && $shipment->has_tracking(); +$shipment_items = $shipment->get_items(); + +if ( 'return' === $shipment->get_type() ) { + if ( $provider = $shipment->get_shipping_provider_instance() ) { + if ( $provider->hide_return_address() ) { + $show_receiver_details = false; + } + } +} +?> +
+ + +

+ + + + + + + + + + + + $item ) { + $product = $item->get_product(); + + wc_get_template( + 'shipment/shipment-details-item.php', + array( + 'shipment' => $shipment, + 'item_id' => $item_id, + 'item' => $item, + 'product' => $product, + ) + ); + } + + /** + * This action is executed after printing the shipment table items on the customer account page. + * + * @param Shipment $shipment The shipment instance. + * + * @since 3.0.0 + * @package Vendidero/Germanized/Shipments + */ + do_action( 'woocommerce_gzd_shipment_details_after_shipment_table_items', $shipment ); + ?> + +
+ + +
+ + $shipment ) ); +} + +if ( $show_tracking ) { + wc_get_template( 'shipment/shipment-details-tracking.php', array( 'shipment' => $shipment ) ); +} diff --git a/packages/woocommerce-germanized-shipments/templates/shipment/shipment-return-instructions.php b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-return-instructions.php new file mode 100644 index 000000000..624594d02 --- /dev/null +++ b/packages/woocommerce-germanized-shipments/templates/shipment/shipment-return-instructions.php @@ -0,0 +1,41 @@ +get_shipping_provider_instance(); +?> + +has_return_instructions() ) : ?> +
get_return_instructions() ) ) ) ); ?>
+ + +has_status( 'delivered' ) && ( $label = $shipment->get_label() ) ) : ?> +

+ + + diff --git a/packages/woocommerce-germanized-shipments/woocommerce-germanized-shipments.php b/packages/woocommerce-germanized-shipments/woocommerce-germanized-shipments.php new file mode 100644 index 000000000..cb93acfdd --- /dev/null +++ b/packages/woocommerce-germanized-shipments/woocommerce-germanized-shipments.php @@ -0,0 +1,72 @@ + +
+

+ composer install', + '' . esc_html( str_replace( ABSPATH, '', __DIR__ ) ) . '' + ); + ?> +

+
+ 0 ) { + if ( $.inArray( elementId, exclude_hide_experts ) == -1 ) { + showElement = false; + } + } + } + + if ( showElement ) { + $( this ).parents( 'tr' ).show(); + } else { + $( this ).parents( 'tr' ).hide(); + } + }); + }, + + onSidebarTitelChange: function() { + var $next = $( this ).nextAll( 'table.form-table:first' ); + $next.find( 'tr:first' ).trigger( 'click' ); + }, + + onSidebarChange: function() { + var $sidebar_elem = $( this ).find( '[data-sidebar]' ), + $table = $( this ).parents( '.form-table' ), + $current_sidebar = $( '.wc-ts-sidebar-active' ), + $sidebar = $current_sidebar; + + if ( $sidebar_elem.length <= 0 ) { + if ( $table.find( '[data-sidebar]' ).length > 0 ) { + $sidebar_elem = $table.find( '[data-sidebar]:first' ); + } + } + + if ( $sidebar_elem.length <= 0 ) { + $sidebar = $( '#wc-ts-sidebar-default' ); + } else { + $sidebar = $( '#' + $sidebar_elem.data( 'sidebar' ) ); + } + + $current_sidebar.removeClass( 'wc-ts-sidebar-active' ); + $sidebar.addClass( 'wc-ts-sidebar-active' ); + }, + + getSettingsWrapper: function() { + var self = trusted_shops.admin; + var prefix = self.optionPrefix.replace( '_', '-' ); + + return $( '.wc-' + prefix + 'admin-settings' ); + }, + + addNotice: function( type, texts ) { + var self = trusted_shops.admin; + + self.getSettingsWrapper().find( '#message' ).remove(); + self.getSettingsWrapper().prepend( '

' + texts.join( '
' ) + '

' ); + + $( 'html, body' ).animate( { + scrollTop: ( self.getSettingsWrapper().offset().top - 100 ) + }, 1000 ); + }, + + validate: function( $elem ) { + var self = trusted_shops.admin, + isValid = true, + id = $elem.attr( 'id' ), + isCode = id.substr( id.length - 5 ) === '_code', + value = $elem.val(); + + if ( $elem.data( 'validate' ) ) { + var type = $elem.data( 'validate' ); + + if ( 'integer' === type ) { + value = parseInt( value ); + + if ( isNaN( value ) ) { + isValid = false; + } + } + } else if( self.isExpertMode() && isCode ) { + if ( '' === value ) { + isValid = false; + } + } + + return isValid; + }, + + onSaveForm: function() { + var self = trusted_shops.admin; + var doSubmit = true; + + $( 'textarea, input, select' ).removeClass( 'wc-ts-has-error' ); + + $( 'textarea:visible, input:visible, select:visible' ).each( function() { + + var id = $( this ).attr( 'id' ), + isCode = id.substr( id.length - 5 ) === '_code', + $td = $( this ).parents( 'tr' ).find( 'td' ); + + $td.find( '.wc-ts-error' ).remove(); + + if ( ! self.validate( $( this ) ) ) { + $( this ).addClass( 'wc-ts-has-error' ); + + if ( isCode ) { + var message = self.params.i18n_error_mandatory; + } else { + var message = $( this ).data( 'validate-msg' ); + } + + $td.append( '' + message + '' ); + + doSubmit = false; + } + }); + + if ( ! doSubmit ) { + $( 'html, body' ).animate( { + scrollTop: ( self.getSettingsWrapper().find( '.wc-ts-has-error:first' ).offset().top - 100 ) + }, 1000 ); + } + + return doSubmit; + }, + + isExpertMode: function() { + var self = trusted_shops.admin; + return $( '#woocommerce_' + this.optionPrefix + 'trusted_shops_integration_mode' ).val() === 'expert'; + }, + + onClickExport: function() { + var self = trusted_shops.admin; + var href_org = $( this ).data( 'href-org' ); + + $( this ).attr( 'href', href_org + '&interval=' + $( '#woocommerce_' + self.optionPrefix + 'trusted_shops_review_collector' ).val() + '&days=' + $( '#woocommerce_' + self.params.option_prefix + 'trusted_shops_review_collector_days_to_send' ).val() ); + } + }; + + $( document ).ready( function() { + trusted_shops.admin.init(); + }); + +})( jQuery, wp, window.trusted_shops ); \ No newline at end of file diff --git a/packages/woocommerce-trusted-shops/assets/js/admin.min.js b/packages/woocommerce-trusted-shops/assets/js/admin.min.js new file mode 100644 index 000000000..6d3e2bf56 --- /dev/null +++ b/packages/woocommerce-trusted-shops/assets/js/admin.min.js @@ -0,0 +1 @@ +window.trusted_shops=window.trusted_shops||{},function(s,a){a.admin={params:{},optionPrefix:"",init:function(){this.params=trusted_shops_params,this.optionPrefix=this.params.option_prefix;var e=this;s(document).on("click","a.woocommerce-ts-input-toggle-trigger",this.onInputToogleClick),s(document).on("change","#woocommerce_"+this.optionPrefix+"trusted_shops_integration_mode",this.onChangeIntegrationMode),s(document).on("change",":input[id$=_enable]",this.onChangeEnable),s(document).on("change","#woocommerce_"+this.optionPrefix+"trusted_shops_reviews_enable",this.onChangeEnableReviews),s(document).find("#woocommerce_"+this.optionPrefix+"trusted_shops_integration_mode").trigger("change"),s(document).find(":input[id$=_enable]").trigger("change"),s(document).on("click","#wc-gzd-trusted-shops-export",this.onClickExport),s(document).on("click","table.form-table tr",this.onSidebarChange),s(":data(sidebar)").each(function(){s(this).parents("tr").on("click",e.onSidebarChange)}),s(document).on("click",'h2, div[id$="options-description"]',this.onSidebarTitelChange),s(document).on("submit","#mainform",this.onSaveForm)},onInputToogleClick:function(){var e=s(this).find("span.woocommerce-ts-input-toggle"),t=e.parents("tr").find("input[type=checkbox]"),o=e.hasClass("woocommerce-input-toggle--enabled");return e.removeClass("woocommerce-input-toggle--enabled"),e.removeClass("woocommerce-input-toggle--disabled"),o?(t.prop("checked",!1),e.addClass("woocommerce-input-toggle--disabled")):(t.prop("checked",!0),e.addClass("woocommerce-input-toggle--enabled")),t.trigger("change"),!1},onChangeEnableReviews:function(){var e=a.admin;s(this).is(":checked")?(s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_sticker_enable").parents("tr").show(),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_widget_enable").parents("tr").show(),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_brand_attribute").parents("tr").show()):(s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_sticker_enable").prop("checked",!1),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_widget_enable").prop("checked",!1),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_sticker_enable").parents("tr").hide(),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_widget_enable").parents("tr").hide(),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_brand_attribute").parents("tr").hide()),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_sticker_enable").trigger("change"),s(document).find("#woocommerce_"+e.optionPrefix+"trusted_shops_product_widget_enable").trigger("change")},onChangeIntegrationMode:function(){a.admin;s(document).find(":input[id$=_enable]").trigger("change")},onChangeEnable:function(){(self=a.admin).showHideGroupElements(s(this))},showHideGroupElements:function(e){var t=e.attr("id"),o=a.admin,t=t.replace("woocommerce_"+o.optionPrefix+"trusted_shops_",""),i=t.substr(0,t.length-7),t=s(":input[id^=woocommerce_"+o.optionPrefix+"trusted_shops_"+i+"_], th[id^=woocommerce_"+o.optionPrefix+"trusted_shops_"+i+"_]"),r=!1,n=["woocommerce_"+o.optionPrefix+"trusted_shops_rich_snippets_category","woocommerce_"+o.optionPrefix+"trusted_shops_rich_snippets_product","woocommerce_"+o.optionPrefix+"trusted_shops_rich_snippets_home","woocommerce_"+o.optionPrefix+"trusted_shops_product_sticker_tab_text"];e.is(":checked")&&(r=!0),t.each(function(){var e=s(this).attr("id"),t=r;"woocommerce_"+o.optionPrefix+"trusted_shops_"+i+"_enable"!==e&&("woocommerce_"+o.optionPrefix+"trusted_shops_"+i+"_code"===e||"woocommerce_"+o.optionPrefix+"trusted_shops_"+i+"_selector"===e?!o.isExpertMode()&&t&&(t=!1):o.isExpertMode()&&0

'+t.join("
")+"

"),s("html, body").animate({scrollTop:o.getSettingsWrapper().offset().top-100},1e3)},validate:function(e){var t=a.admin,o=!0,i=e.attr("id"),i="_code"===i.substr(i.length-5),r=e.val();return e.data("validate")?"integer"===e.data("validate")&&(r=parseInt(r),isNaN(r)&&(o=!1)):t.isExpertMode()&&i&&""===r&&(o=!1),o},onSaveForm:function(){var o=a.admin,i=!0;return s("textarea, input, select").removeClass("wc-ts-has-error"),s("textarea:visible, input:visible, select:visible").each(function(){var e=s(this).attr("id"),e="_code"===e.substr(e.length-5),t=s(this).parents("tr").find("td");t.find(".wc-ts-error").remove(),o.validate(s(this))||(s(this).addClass("wc-ts-has-error"),e=e?o.params.i18n_error_mandatory:s(this).data("validate-msg"),t.append(''+e+""),i=!1)}),i||s("html, body").animate({scrollTop:o.getSettingsWrapper().find(".wc-ts-has-error:first").offset().top-100},1e3),i},isExpertMode:function(){a.admin;return"expert"===s("#woocommerce_"+this.optionPrefix+"trusted_shops_integration_mode").val()},onClickExport:function(){var e=a.admin,t=s(this).data("href-org");s(this).attr("href",t+"&interval="+s("#woocommerce_"+e.optionPrefix+"trusted_shops_review_collector").val()+"&days="+s("#woocommerce_"+e.params.option_prefix+"trusted_shops_review_collector_days_to_send").val())}},s(document).ready(function(){a.admin.init()})}(jQuery,(wp,window.trusted_shops)); \ No newline at end of file diff --git a/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE.mo b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE.mo new file mode 100644 index 000000000..06afcb3b9 Binary files /dev/null and b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE.mo differ diff --git a/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE.po b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE.po new file mode 100644 index 000000000..56fe16296 --- /dev/null +++ b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE.po @@ -0,0 +1,1166 @@ +msgid "" +msgstr "" +"Project-Id-Version: WooCommerce Trusted Shops\n" +"POT-Creation-Date: 2019-10-30 17:53+0100\n" +"PO-Revision-Date: 2019-10-30 19:17+0100\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: de_DE\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"X-Generator: Poedit 2.2.4\n" +"X-Poedit-Basepath: ../..\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Poedit-KeywordsList: __;_e;__ngettext:1,2;_n:1,2;__ngettext_noop:1,2;" +"_n_noop:1,2;_c,_nc:4c,1,2;_x:1,2c;_ex:1,2c;_nx:4c,1,2;_nx_noop:4c,1,2;" +"esc_attr_e;esc_html_e;esc_html__\n" +"X-Poedit-SearchPath-0: .\n" +"X-Poedit-SearchPathExcluded-0: node_modules\n" +"X-Poedit-SearchPathExcluded-1: vendor\n" + +#: includes/admin/settings/class-wc-ts-gzd-settings-tab.php:48 +msgctxt "trusted-shops" +msgid "Setup your Trusted Shops Integration." +msgstr "Setze deine Trusted Shops Integration auf." + +#: includes/admin/settings/class-wc-ts-gzd-settings-tab.php:52 +#: includes/class-wc-ts-settings-handler.php:23 +msgctxt "trusted-shops" +msgid "Trusted Shops" +msgstr "Trusted Shops" + +#: includes/admin/views/html-notice-dependencies.php:16 +msgctxt "trusted-shops" +msgid "Dependencies Missing or Outdated" +msgstr "Wichtige Plugins fehlen oder sind veraltet" + +#: includes/admin/views/html-notice-dependencies.php:24 +msgctxt "trusted-shops" +msgid "" +"To use WooCommerce Trusted Shops you may at first install the following " +"plugins:" +msgstr "" +"Um WooCommerce Trusted Shops zuverlässig nutzen zu können, musst du erst " +"folgende Plugins installieren:" + +#: includes/admin/views/html-notice-dependencies.php:28 +#, php-format +msgctxt "trusted-shops" +msgid "Install %s" +msgstr "Install %s" + +#: includes/admin/views/html-notice-dependencies.php:36 +msgctxt "trusted-shops" +msgid "" +"To use WooCommerce Trusted Shops you may at first update the following " +"plugins to a newer version:" +msgstr "" +"Um WooCommerce Trusted Shops zuverlässig nutzen zu können, update bitte " +"folgende Plugins:" + +#: includes/admin/views/html-notice-dependencies.php:40 +#, php-format +msgctxt "trusted-shops" +msgid "%s required in at least version %s" +msgstr "%s wird mindestens in Version %s benötigt" + +#: includes/admin/views/html-notice-dependencies.php:49 +msgctxt "trusted-shops" +msgid "Check for Updates" +msgstr "nach Updates suchen" + +#: includes/admin/views/html-notice-dependencies.php:50 +msgctxt "trusted-shops" +msgid "or" +msgstr "oder" + +#: includes/admin/views/html-notice-dependencies.php:51 +msgctxt "trusted-shops" +msgid "Install an older version" +msgstr "Installiere eine ältere Version" + +#: includes/admin/views/html-notice-update.php:14 +msgctxt "trusted-shops" +msgid "" +"WooCommerce Trusted Shops Data Update Required – We " +"just need to update your installation to the latest version" +msgstr "" +"WooCommerce Trusted Shops Datenaktualisierung erforderlich " +"– Wir müssen deine Installation auf die neueste Version updaten" + +#: includes/admin/views/html-notice-update.php:15 +msgctxt "trusted-shops" +msgid "Run the updater" +msgstr "Update starten" + +#: includes/admin/views/html-notice-update.php:19 +msgctxt "trusted-shops" +msgid "" +"It is strongly recommended that you backup your database before proceeding. " +"Are you sure you wish to run the updater now?" +msgstr "" +"Du solltest vor einem Update immer ein Backup deiner Datenbank anlegen. Bist " +"du sicher das Update jetzt zu installieren?" + +#: includes/admin/views/html-wpml-notice.php:11 +msgctxt "trusted-shops" +msgid "WPML Support" +msgstr "WPML Unterstützung" + +#: includes/admin/views/html-wpml-notice.php:14 +msgctxt "trusted-shops" +msgid "" +"These settings serve as default settings for all your languages. To adjust " +"the settings for a certain language, please switch your admin language " +"through the WPML language switcher and adjust the corresponding settings." +msgstr "" +"Diese Einstellungen werden als Standard für alle deine Sprachen verwendet. " +"Um die Einstellungen für eine spezielle Sprache zu ändern, wechsle bitte zur " +"entsprechenden Sprache über den WPML Sprachumschalter." + +#: includes/admin/views/html-wpml-notice.php:16 +#, php-format +msgctxt "trusted-shops" +msgid "" +"These settings apply for your %s shop. To adjust settings for another " +"language, please switch your admin language through the WPML language " +"switcher." +msgstr "" +"Diese Einstellungen werden für deinen %s Shop verwendet. Um die " +"Einstellungen für eine andere Sprache anzupassen, wechsle bitte die Sprache " +"über den WPML Sprachumschalter." + +#: includes/class-wc-trusted-shops-admin.php:84 +#: includes/class-wc-trusted-shops-admin.php:121 +msgctxt "trusted-shops" +msgid "GTIN" +msgstr "GTIN" + +#: includes/class-wc-trusted-shops-admin.php:84 +#: includes/class-wc-trusted-shops-admin.php:121 +msgctxt "trusted-shops" +msgid "" +"ID that allows your products to be identified worldwide. If you want to " +"display your Trusted Shops Product Reviews in Google Shopping and paid " +"Google adverts, Google needs the GTIN." +msgstr "" +"Identifikationsnummer, mit der Produkte weltweit eindeutig identifiziert " +"werden können. Wenn du deine Trusted Shops Produktbewertungen in Google " +"Shopping und bezahlten Google Produktanzeigen ausspielen möchtest, benötigt " +"Google die GTIN." + +#: includes/class-wc-trusted-shops-admin.php:88 +#: includes/class-wc-trusted-shops-admin.php:122 +msgctxt "trusted-shops" +msgid "MPN" +msgstr "MPN" + +#: includes/class-wc-trusted-shops-admin.php:88 +#: includes/class-wc-trusted-shops-admin.php:122 +msgctxt "trusted-shops" +msgid "" +"If you don't have a GTIN for your products, you can pass the brand name and " +"the MPN on to Google to use the Trusted Shops Google Integration." +msgstr "" +"Wenn deine Produkte keine GTIN haben, kannst du den Markennamen zusammen mit " +"der MPN übergeben, um die Trusted Shops Google Integration zu nutzen." + +#: includes/class-wc-trusted-shops-admin.php:142 +msgctxt "trusted-shops" +msgid "Brand" +msgstr "Marke" + +#: includes/class-wc-trusted-shops-admin.php:176 +msgctxt "trusted-shops" +msgid "This field is mandatory" +msgstr "Dieses Feld ist ein Pflichtfeld" + +#: includes/class-wc-trusted-shops-admin.php:183 +msgctxt "trusted-shops" +msgid "Trusted Shops Options" +msgstr "Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:190 +msgctxt "trusted-shops" +msgid "Arial" +msgstr "Arial" + +#: includes/class-wc-trusted-shops-admin.php:191 +msgctxt "trusted-shops" +msgid "Geneva" +msgstr "Geneva" + +#: includes/class-wc-trusted-shops-admin.php:192 +msgctxt "trusted-shops" +msgid "Georgia" +msgstr "Georgia" + +#: includes/class-wc-trusted-shops-admin.php:193 +msgctxt "trusted-shops" +msgid "Helvetica" +msgstr "Helvetica" + +#: includes/class-wc-trusted-shops-admin.php:194 +msgctxt "trusted-shops" +msgid "Sans-serif" +msgstr "Sans-serif" + +#: includes/class-wc-trusted-shops-admin.php:195 +msgctxt "trusted-shops" +msgid "Serif" +msgstr "Serif" + +#: includes/class-wc-trusted-shops-admin.php:196 +msgctxt "trusted-shops" +msgid "Trebuchet MS" +msgstr "Trebuchet MS" + +#: includes/class-wc-trusted-shops-admin.php:197 +msgctxt "trusted-shops" +msgid "Verdana" +msgstr "Verdana" + +#: includes/class-wc-trusted-shops-admin.php:220 +msgctxt "trusted-shops" +msgid "Trusted Shops Integration" +msgstr "Trusted Shops Integration" + +#: includes/class-wc-trusted-shops-admin.php:221 +#, php-format +msgctxt "trusted-shops" +msgid "Do you need help with integrating your Trustbadge? %s" +msgstr "Brauchst du Hilfe bei der Einbindung deines Trustbadges? %s" + +#: includes/class-wc-trusted-shops-admin.php:221 +msgctxt "trusted-shops" +msgid "To the step-by-step instructions" +msgstr "Zur Schritt-für-Schritt Anleitung" + +#: includes/class-wc-trusted-shops-admin.php:227 +msgctxt "trusted-shops" +msgid "Trusted Shops ID" +msgstr "Trusted Shops ID" + +#: includes/class-wc-trusted-shops-admin.php:228 +msgctxt "trusted-shops" +msgid "" +"The Trusted Shops ID is a unique identifier for your shop. You can find your " +"Trusted Shops ID in your My Trusted Shops account." +msgstr "" +"Die Trusted Shops ID identifiziert deinen Shop eindeutig bei Trusted Shops. " +"Du findest deine Trusted Shops ID in deinem My Trusted Shops Account." + +#: includes/class-wc-trusted-shops-admin.php:237 +msgctxt "trusted-shops" +msgid "Edit Mode" +msgstr "Bearbeitungsmodus" + +#: includes/class-wc-trusted-shops-admin.php:239 +#: includes/class-wc-trusted-shops-admin.php:300 +#: includes/class-wc-trusted-shops-admin.php:423 +msgctxt "trusted-shops" +msgid "" +"The advanced configuration is for users with programming skills. Here you " +"can create even more individual settings." +msgstr "" +"Der Expertenmodus ist für fortgeschrittene Nutzer mit Programmierkenntnissen " +"gedacht. Hier kannst du noch mehr individuelle Einstellungen vornehmen." + +#: includes/class-wc-trusted-shops-admin.php:243 +msgctxt "trusted-shops" +msgid "Standard configuration" +msgstr "Standardmodus" + +#: includes/class-wc-trusted-shops-admin.php:244 +msgctxt "trusted-shops" +msgid "Advanced configuration" +msgstr "Expertenmodus" + +#: includes/class-wc-trusted-shops-admin.php:252 +msgctxt "trusted-shops" +msgid "Configure your Trustbadge" +msgstr "Konfiguriere dein Trustbadge" + +#: includes/class-wc-trusted-shops-admin.php:258 +msgctxt "trusted-shops" +msgid "Display Trustbadge" +msgstr "Trustbadge anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:260 +msgctxt "trusted-shops" +msgid "Display the Trustbadge on all the pages of your shop." +msgstr "Zeige das Trustbadge auf allen Seiten deines Shops an." + +#: includes/class-wc-trusted-shops-admin.php:267 +msgctxt "trusted-shops" +msgid "Variant" +msgstr "Variante" + +#: includes/class-wc-trusted-shops-admin.php:269 +msgctxt "trusted-shops" +msgid "You can display your Trustbadge with or without Review Stars." +msgstr "Du kannst dein Trustbadge mit oder ohne Bewertungssterne anzeigen." + +#: includes/class-wc-trusted-shops-admin.php:273 +#: includes/class-wc-trusted-shops-admin.php:763 +msgctxt "trusted-shops" +msgid "Display Trustbadge with review stars" +msgstr "Trustbadge mit Bewertungssternen anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:274 +#: includes/class-wc-trusted-shops-admin.php:767 +msgctxt "trusted-shops" +msgid "Display Trustbadge without review stars" +msgstr "Trustbadge ohne Bewertungssterne anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:280 +msgctxt "trusted-shops" +msgid "Vertical Offset" +msgstr "Vertikaler Abstand" + +#: includes/class-wc-trusted-shops-admin.php:281 +msgctxt "trusted-shops" +msgid "" +"Choose the distance that the Trustbadge will appear from the bottom-right " +"corner of the screen." +msgstr "" +"Wähle einen Abstand für dein Trustbadge vom unteren rechten Bildschirmrand." + +#: includes/class-wc-trusted-shops-admin.php:284 +#: includes/class-wc-trusted-shops-admin.php:492 +#: includes/class-wc-trusted-shops-admin.php:543 +#: includes/class-wc-trusted-shops-admin.php:558 +msgctxt "trusted-shops" +msgid "px" +msgstr "px" + +#: includes/class-wc-trusted-shops-admin.php:290 +#: includes/class-wc-trusted-shops-admin.php:499 +#: includes/class-wc-trusted-shops-admin.php:549 +#: includes/class-wc-trusted-shops-admin.php:565 +#, php-format +msgctxt "trusted-shops" +msgid "Please choose a non-negative number (at least %d)" +msgstr "Bitte verwende eine nicht-negative Nummer (mindestens %d)" + +#: includes/class-wc-trusted-shops-admin.php:296 +msgctxt "trusted-shops" +msgid "Trustbadge code" +msgstr "Trustbadge Code" + +#: includes/class-wc-trusted-shops-admin.php:308 +msgctxt "trusted-shops" +msgid "Configure your Shop Reviews" +msgstr "Konfiguriere deine Shopbewertungen" + +#: includes/class-wc-trusted-shops-admin.php:314 +msgctxt "trusted-shops" +msgid "Display Shop Review Sticker" +msgstr "Shopbewertungssticker anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:315 +msgctxt "trusted-shops" +msgid "" +"To display the Shop Review Sticker, you have to assign the widget \"Trusted " +"Shops Review Sticker\"." +msgstr "" +"Um den Shopbewertungssticker anzuzeigen, musst du das Widget „Trusted Shops " +"Shopbewertungssticker“ zuweisen." + +#: includes/class-wc-trusted-shops-admin.php:316 +#, php-format +msgctxt "trusted-shops" +msgid "Assign widget %s" +msgstr "Widget %s zuweisen" + +#: includes/class-wc-trusted-shops-admin.php:316 +#: includes/class-wc-trusted-shops-admin.php:588 +#: includes/class-wc-trusted-shops-admin.php:896 +msgctxt "trusted-shops" +msgid "here" +msgstr "hier" + +#: includes/class-wc-trusted-shops-admin.php:324 +#: includes/class-wc-trusted-shops-admin.php:472 +msgctxt "trusted-shops" +msgid "Background color" +msgstr "Hintergrundfarbe" + +#: includes/class-wc-trusted-shops-admin.php:325 +msgctxt "trusted-shops" +msgid "Choose the background color for your Review Sticker." +msgstr "Wähle die Hintergrundfarbe für deinen Review Sticker." + +#: includes/class-wc-trusted-shops-admin.php:332 +msgctxt "trusted-shops" +msgid "Font" +msgstr "Schriftart" + +#: includes/class-wc-trusted-shops-admin.php:334 +msgctxt "trusted-shops" +msgid "Choose the font for your Review Sticker." +msgstr "Wähle die Schriftart für deinen Review Sticker." + +#: includes/class-wc-trusted-shops-admin.php:341 +msgctxt "trusted-shops" +msgid "Number of reviews displayed" +msgstr "Anzahl Bewertungen" + +#: includes/class-wc-trusted-shops-admin.php:342 +msgctxt "trusted-shops" +msgid "" +"Display x alternating Shop Reviews in your Shop Review Sticker. You can " +"display between 1 and 5 alternating Shop Reviews." +msgstr "" +"Zeige x Shopbewertungen im Wechsel in deinem Shopbewertungssticker an. Es " +"können mindestens 1 und maximal 5 Bewertungen im Wechsel angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:345 +msgctxt "trusted-shops" +msgid "Show x alternating reviews" +msgstr "x Bewertungen im Wechsel zeigen" + +#: includes/class-wc-trusted-shops-admin.php:352 +#: includes/class-wc-trusted-shops-admin.php:369 +#, php-format +msgctxt "trusted-shops" +msgid "Please choose a non-negative number between %d and %d" +msgstr "Bitte verwende eine nicht-negative Nummer zwischen %d und %d" + +#: includes/class-wc-trusted-shops-admin.php:358 +msgctxt "trusted-shops" +msgid "Minimum rating displayed" +msgstr "Angezeigte Mindestnote" + +#: includes/class-wc-trusted-shops-admin.php:359 +msgctxt "trusted-shops" +msgid "Only show Shop Reviews with a minimum rating of x stars. " +msgstr "Zeige nur Shopbewertungen mit einer Mindestanzahl von x Sternen." + +#: includes/class-wc-trusted-shops-admin.php:362 +msgctxt "trusted-shops" +msgid "Star(s)" +msgstr "Stern(e)" + +#: includes/class-wc-trusted-shops-admin.php:375 +msgctxt "trusted-shops" +msgid "Sticker code" +msgstr "Sticker Code" + +#: includes/class-wc-trusted-shops-admin.php:379 +#: includes/class-wc-trusted-shops-admin.php:506 +#: includes/class-wc-trusted-shops-admin.php:571 +msgctxt "trusted-shops" +msgid "" +"The advanced configuration is for users with programming skills. Here you " +"can perform even more individual settings." +msgstr "" +"Der Expertenmodus ist für fortgeschrittene Nutzer mit Programmierkenntnissen " +"gedacht. Hier kannst du noch mehr individuelle Einstellungen vornehmen." + +#: includes/class-wc-trusted-shops-admin.php:385 +msgctxt "trusted-shops" +msgid "Google Organic Search" +msgstr "Organische Google Suche" + +#: includes/class-wc-trusted-shops-admin.php:386 +msgctxt "trusted-shops" +msgid "" +"Activate this option to give Google the opportunity to show your Shop " +"Reviews in Google organic search results." +msgstr "" +"Aktiviere diese Funktion, um Google die Möglichkeit zu geben, deine " +"Shopbewertungen in den organischen Google Suchergebnissen anzuzeigen." + +#: includes/class-wc-trusted-shops-admin.php:387 +msgctxt "trusted-shops" +msgid "" +"By activating this option, rich snippets will be integrated in the selected " +"pages so your shop review stars may be displayed in Google organic search " +"results. If you use Product Reviews and already activated rich snippets in " +"expert mode, we recommend integrating rich snippets for Shop Reviews on " +"category pages only." +msgstr "" +"Wenn du diese Option aktivierst, werden Rich Snippets in die ausgewählten " +"Seiten eingebunden, sodass deine Shopbewertungssterne in den organischen " +"Google Suchergebnissen angezeigt werden können. Wenn du Produktbewertungen " +"aktiviert hast und dort im Expertenmodus bereits Rich Snippets aktiviert " +"hast, empfehlen wir die Ausgabe der Rich Snippets für Shopbewertungen nur " +"auf der Kategorieseite." + +#: includes/class-wc-trusted-shops-admin.php:394 +msgctxt "trusted-shops" +msgid "Activate rich snippets on" +msgstr "Rich Snippets aktivieren auf" + +#: includes/class-wc-trusted-shops-admin.php:395 +msgctxt "trusted-shops" +msgid "category pages" +msgstr "Kategorieseiten" + +#: includes/class-wc-trusted-shops-admin.php:403 +msgctxt "trusted-shops" +msgid "product pages" +msgstr "Produktdetailseiten" + +#: includes/class-wc-trusted-shops-admin.php:411 +msgctxt "trusted-shops" +msgid "homepage (not recommended)" +msgstr "Startseite (nicht empfohlen)" + +#: includes/class-wc-trusted-shops-admin.php:419 +msgctxt "trusted-shops" +msgid "Rich snippets code" +msgstr "Rich Snippets Code" + +#: includes/class-wc-trusted-shops-admin.php:431 +msgctxt "trusted-shops" +msgid "Configure your Product Reviews " +msgstr "Konfiguriere deine Produktbewertungen" + +#: includes/class-wc-trusted-shops-admin.php:432 +#, php-format +msgctxt "trusted-shops" +msgid "To use Product Reviews, activate them in your %s first." +msgstr "" +"Um Produktbewertungen nutzen zu können, schalte diese zuerst in deinem %s " +"frei." + +#: includes/class-wc-trusted-shops-admin.php:432 +msgctxt "trusted-shops" +msgid "Trusted Shops package" +msgstr "Trusted Shops Paket" + +#: includes/class-wc-trusted-shops-admin.php:438 +msgctxt "trusted-shops" +msgid "Collect Product Reviews" +msgstr "Produktbewertungen sammeln" + +#: includes/class-wc-trusted-shops-admin.php:439 +msgctxt "trusted-shops" +msgid "" +"Show Product Reviews on the product page in a separate tab, just as shown on " +"the picture on the right." +msgstr "" +"Zeige Produktbewertungen auf der Produktseite in einem separaten Reiter, wie " +"in der Grafik rechts abgebildet." + +#: includes/class-wc-trusted-shops-admin.php:447 +msgctxt "trusted-shops" +msgid "Reviews" +msgstr "Bewertungen" + +#: includes/class-wc-trusted-shops-admin.php:448 +#: includes/class-wc-trusted-shops-admin.php:457 +msgctxt "trusted-shops" +msgid "You can choose a name for the tab with your Product Reviews." +msgstr "" +"Du kannst selbst bestimmen, wie der Reiter in dem deine Produktbewertungen " +"angezeigt werden, heißen soll." + +#: includes/class-wc-trusted-shops-admin.php:449 +msgctxt "trusted-shops" +msgid "Show Product Reviews on the product detail page in an additional tab." +msgstr "Produktbewertungen auf der Produktseite in separatem Reiter anzeigen." + +#: includes/class-wc-trusted-shops-admin.php:456 +msgctxt "trusted-shops" +msgid "Name of Product Reviews tab" +msgstr "Bezeichnung Reiter" + +#: includes/class-wc-trusted-shops-admin.php:460 +msgctxt "trusted-shops" +msgid "Product reviews" +msgstr "Produktbewertungen" + +#: includes/class-wc-trusted-shops-admin.php:464 +msgctxt "trusted-shops" +msgid "Border color" +msgstr "Umrandung" + +#: includes/class-wc-trusted-shops-admin.php:465 +msgctxt "trusted-shops" +msgid "Set the color for the frame around your Product Reviews." +msgstr "Lege die Farbe für den Rahmen um deine Produktbewertungen fest." + +#: includes/class-wc-trusted-shops-admin.php:473 +msgctxt "trusted-shops" +msgid "Set the background color for your Product Reviews." +msgstr "Lege die Hintergrundfarbe für deine Produktbewertungen fest." + +#: includes/class-wc-trusted-shops-admin.php:480 +#: includes/class-wc-trusted-shops-admin.php:530 +msgctxt "trusted-shops" +msgid "Star color" +msgstr "Farbe der Sterne" + +#: includes/class-wc-trusted-shops-admin.php:481 +msgctxt "trusted-shops" +msgid "Set the color for the Product Review stars in your Product Reviews tab." +msgstr "" +"Lege die Farbe der Sterne fest, die in deinem Produktbewertungs-Reiter " +"angezeigt werden. " + +#: includes/class-wc-trusted-shops-admin.php:488 +#: includes/class-wc-trusted-shops-admin.php:538 +msgctxt "trusted-shops" +msgid "Star size" +msgstr "Größe der Sterne" + +#: includes/class-wc-trusted-shops-admin.php:493 +msgctxt "trusted-shops" +msgid "Set the size for the Product Review stars in your Product Reviews tab." +msgstr "" +"Lege die Größe der Sterne fest, die in deinem Produktbewertungs-Reiter " +"angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:504 +msgctxt "trusted-shops" +msgid "Product Sticker Code" +msgstr "Produktbewertungen Code" + +#: includes/class-wc-trusted-shops-admin.php:513 +#: includes/class-wc-trusted-shops-admin.php:579 +msgctxt "trusted-shops" +msgid "jQuerySelector" +msgstr "jQuerySelector" + +#: includes/class-wc-trusted-shops-admin.php:514 +msgctxt "trusted-shops" +msgid "" +"Please choose where your Product Reviews shall be displayed on the Product " +"detail page." +msgstr "" +"Wähle, wo die Produktbewertungen auf der Produktdetailseite angezeigt werden " +"sollen." + +#: includes/class-wc-trusted-shops-admin.php:521 +msgctxt "trusted-shops" +msgid "Rating stars" +msgstr "Bewertungssterne" + +#: includes/class-wc-trusted-shops-admin.php:522 +msgctxt "trusted-shops" +msgid "Show star ratings on the product detail page below your product name." +msgstr "Bewertungssterne auf der Produktseite unter dem Produktnamen anzeigen." + +#: includes/class-wc-trusted-shops-admin.php:523 +msgctxt "trusted-shops" +msgid "" +"Display Product Review stars on product pages below the product name, just " +"as shown in the picture on the right." +msgstr "" +"Zeige Bewertungssterne auf der Produktseite unter dem Produktnamen, wie in " +"der Grafik rechts abgebildet." + +#: includes/class-wc-trusted-shops-admin.php:532 +msgctxt "trusted-shops" +msgid "" +"Set the color for the review stars, that are displayed on the product page, " +"below your product name." +msgstr "" +"Lege die Farbe der Sterne fest, die auf der Produktseite unter deinem " +"Produktnamen angezeigt werden. " + +#: includes/class-wc-trusted-shops-admin.php:540 +msgctxt "trusted-shops" +msgid "" +"Set the size for the review stars that are displayed on the product page, " +"below your product name." +msgstr "" +"Lege die Größe der Sterne fest, die auf der Produktseite unter deinem " +"Produktnamen angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:554 +msgctxt "trusted-shops" +msgid "Font size" +msgstr "Schriftgröße" + +#: includes/class-wc-trusted-shops-admin.php:556 +msgctxt "trusted-shops" +msgid "Set the font size for the text that goes with your review stars." +msgstr "Lege die Schriftgröße für den Text zu deinen Bewertungssternen fest." + +#: includes/class-wc-trusted-shops-admin.php:570 +msgctxt "trusted-shops" +msgid "Product Review Code" +msgstr "Produktbewertungen Code" + +#: includes/class-wc-trusted-shops-admin.php:580 +msgctxt "trusted-shops" +msgid "" +"Please choose where your Product Review Stars shall be displayed on the " +"Product Detail page." +msgstr "" +"Wähle, wo die Produktbewertungssterne auf der Produktdetailseite angezeigt " +"werden sollen." + +#: includes/class-wc-trusted-shops-admin.php:587 +msgctxt "trusted-shops" +msgid "Brand attribute" +msgstr "Marken-Produktattribut" + +#: includes/class-wc-trusted-shops-admin.php:588 +#, php-format +msgctxt "trusted-shops" +msgid "Create brand attribute %s" +msgstr "Marken-Produktattribut %s erstellen" + +#: includes/class-wc-trusted-shops-admin.php:589 +msgctxt "trusted-shops" +msgid "" +"Brand name of the product. By passing this information on to Google, you " +"improve your chances of having Google identify your products. Assign your " +"brand attribute. If your products don't have a GTIN, you can pass on the " +"brand name and the MPN to use Google Integration." +msgstr "" +"Markenname des Produkts. Durch Übergeben dieser Variable verbesserst du die " +"Möglichkeit deine Produkte von Google eindeutig identifizieren zu lassen. " +"Weise dein Marken-Produktattribut zu. Wenn deine Produkte keine GTIN haben, " +"kannst du den Markennamen zusammen mit der MPN übergeben, um die Google " +"Integration zu nutzen." + +#: includes/class-wc-trusted-shops-admin.php:595 +msgctxt "trusted-shops" +msgid "None" +msgstr "Keine" + +#: includes/class-wc-trusted-shops-admin.php:606 +msgctxt "trusted-shops" +msgid "Configure your Review Requests" +msgstr "Konfiguriere deine Bewertungserinnerungen" + +#: includes/class-wc-trusted-shops-admin.php:607 +msgctxt "trusted-shops" +msgid "" +"7 days after an order has been placed, Trusted Shops automatically sends an " +"invite to your customers. If you want to set a different time for sending " +"automatic Review Requests, please activate the option below. If you want to " +"send review requests with legal certainty, you need your customers' consent " +"to receive Review Requests. You also have to include an option to " +"unsubscribe." +msgstr "" +"Trusted Shops versendet 7 Tage nach Bestellung automatisch eine " +"Bewertungserinnerung an deinen Kunden. Wenn du lieber selbst entscheiden " +"möchtest, wann Bewertungsanfragen versendet werden, aktiviere die " +"untenstehende Option. Möchtest du rechtssicher Bewertungs-Erinnerungen " +"verschicken, so benötigst du die ausdrückliche Einwilligung deiner Kunden " +"zum Empfang von Bewertungsemails. Du musst deinen Kunden zudem eine " +"Möglichkeit geben, Bewertungserinnerungen wieder abzubestellen." + +#: includes/class-wc-trusted-shops-admin.php:613 +msgctxt "trusted-shops" +msgid "Enable Review Requests" +msgstr "Bewertungserinnerungen aktivieren" + +#: includes/class-wc-trusted-shops-admin.php:622 +msgctxt "trusted-shops" +msgid "WooCommerce status" +msgstr "Bestellstatus" + +#: includes/class-wc-trusted-shops-admin.php:623 +msgctxt "trusted-shops" +msgid "" +"We recommend choosing the order status that you set when your products have " +"been shipped." +msgstr "" +"Wähle hier am besten den Bestellstatus aus WooCommerce aus, den du " +"einstellst, wenn deine Ware versendet wurde." + +#: includes/class-wc-trusted-shops-admin.php:631 +msgctxt "trusted-shops" +msgid "Days until Review Request" +msgstr "Tage bis zur Erinnerung" + +#: includes/class-wc-trusted-shops-admin.php:632 +msgctxt "trusted-shops" +msgid "" +"Set the number of days to wait after an order has reached the order status " +"you selected above before having a review request sent to your customers." +msgstr "" +"Stelle hier ein nach wie vielen Tagen nach Erreichen des oben ausgewählten " +"Bestellstatus die Bewertungserinnerung an den Käufer versendet werden soll." + +#: includes/class-wc-trusted-shops-admin.php:644 +msgctxt "trusted-shops" +msgid "Permission via checkbox" +msgstr "Einwilligung per Checkbox" + +#: includes/class-wc-trusted-shops-admin.php:645 +msgctxt "trusted-shops" +msgid "" +"If the checkbox is activated, only customers who gave their consent will " +"receive Review Requests." +msgstr "" +"Bei aktivierter Checkbox erhalten nur die Kunden eine Bewertungserinnerung, " +"die ihr Einverständnis gegeben haben." + +#: includes/class-wc-trusted-shops-admin.php:649 +msgctxt "trusted-shops" +msgid "Edit checkbox" +msgstr "Checkbox anpassen" + +#: includes/class-wc-trusted-shops-admin.php:653 +msgctxt "trusted-shops" +msgid "Unsubscribe via link" +msgstr "Abmeldung per Link" + +#: includes/class-wc-trusted-shops-admin.php:654 +msgctxt "trusted-shops" +msgid "Allows the customer to unsubscribe from Review Requests." +msgstr "" +"Erlaubt es dem Kunden sich von Bewertungserinnerungen per Link abzumelden." + +#: includes/class-wc-trusted-shops-admin.php:755 +msgctxt "trusted-shops" +msgid "How does Trusted Shops make your shop better?" +msgstr "Wie macht Trusted Shops deinen Shop besser?" + +#: includes/class-wc-trusted-shops-admin.php:757 +msgctxt "trusted-shops" +msgid "Get your account" +msgstr "Erstelle deinen Account" + +#: includes/class-wc-trusted-shops-admin.php:777 +msgctxt "trusted-shops" +msgid "Product Reviews on the product detail page in an additional tab" +msgstr "Produktbewertungen auf der Produktseite in einem separatem Reiter" + +#: includes/class-wc-trusted-shops-admin.php:780 +msgctxt "trusted-shops" +msgid "Show Star-Ratings on the product detail page below your product name" +msgstr "Bewertungssterne unter dem Produktnamen auf der Produktseite" + +#: includes/class-wc-trusted-shops-admin.php:784 +msgctxt "trusted-shops" +msgid "" +"Please note: If you want to send review requests through WooCommerce, you " +"should deactivate automated review requests through Trusted Shops. To do so, " +"please go to your My Trusted Shops account. Log in and go to Reviews > " +"Settings and deactivate \"Collect reviews automatically\"" +msgstr "" +"Bitte beachte: Wenn du Bewertungserinnerungen über WooCommerce versenden " +"möchtest, solltest du den Versand von automatischen Bewertungserinnerungen " +"über Trusted Shops deaktivieren. Gehe dazu in deinen My Trusted Shops " +"Account. Logge dich ein und gehe zu Bewertungen > Konfiguration und " +"deaktiviere dort „Bewertungen automatisch sammeln“." + +#: includes/class-wc-trusted-shops-admin.php:785 +msgctxt "trusted-shops" +msgid "To your My Trusted Shops account" +msgstr "Zu deinem My Trusted Shops Account" + +#: includes/class-wc-trusted-shops-admin.php:789 +msgctxt "trusted-shops" +msgid "" +"Export your customer information here and upload it in the Trusted Shops " +"Review Collector. To do so go to your My Trusted Shops account. Log in and " +"go to Reviews > Shop Reviews > Review Collector" +msgstr "" +"Exportiere hier die Kundendaten und lade diese im Trusted Shops Review " +"Collector hoch. Gehe dazu in deinen My Trusted Shops Account. Logge dich ein " +"und gehe zu Bewertungen > Shopbewertungen > Review Collector" + +#: includes/class-wc-trusted-shops-admin.php:790 +msgctxt "trusted-shops" +msgid "To the Trusted Shops Review Collector" +msgstr "Zum Trusted Shops Review Collector" + +#: includes/class-wc-trusted-shops-admin.php:880 +msgctxt "trusted-shops" +msgid "Review Collector" +msgstr "Review Collector" + +#: includes/class-wc-trusted-shops-admin.php:882 +#, php-format +msgctxt "trusted-shops" +msgid "" +"Want to collect reviews for orders that were placed before your Trusted " +"Shops Integration? No problem. Export old orders here and upload them in " +"your %s." +msgstr "" +"Du möchtest nachträglich Bewertungen zu Bestellungen sammeln, die vor der " +"Trusted Shops Integration getätigt wurden? Kein Problem. Exportiere alte " +"Bestellungen hier und lade diese in deinem %s hoch." + +#: includes/class-wc-trusted-shops-admin.php:882 +msgctxt "trusted-shops" +msgid "My Trusted Shops account" +msgstr "My Trusted Shops Account" + +#: includes/class-wc-trusted-shops-admin.php:888 +msgctxt "trusted-shops" +msgid "Export orders" +msgstr "Bestellungen exportieren" + +#: includes/class-wc-trusted-shops-admin.php:888 +msgctxt "trusted-shops" +msgid "" +"Export your customer and order information of the last x days and upload " +"them in your My Trusted Shops Account." +msgstr "" +"Exportiere deine Kunden- und Bestelldaten der letzten x Tage und lade diese " +"in deinem My Trusted Shops Account hoch." + +#: includes/class-wc-trusted-shops-admin.php:892 +msgctxt "trusted-shops" +msgid "30 days" +msgstr "30 Tage" + +#: includes/class-wc-trusted-shops-admin.php:893 +msgctxt "trusted-shops" +msgid "60 days" +msgstr "60 Tage" + +#: includes/class-wc-trusted-shops-admin.php:894 +msgctxt "trusted-shops" +msgid "90 days" +msgstr "90 Tage" + +#: includes/class-wc-trusted-shops-admin.php:896 +#, php-format +msgctxt "trusted-shops" +msgid "Upload customer and order information %s." +msgstr "Kunden- und Bestelldaten %s hochladen" + +#: includes/class-wc-trusted-shops-admin.php:899 +msgctxt "trusted-shops" +msgid "Days until reminder mail" +msgstr "Tage bis zur Erinnerung" + +#: includes/class-wc-trusted-shops-admin.php:899 +msgctxt "trusted-shops" +msgid "" +"Set the number of days to wait after the order date before having a Review " +"Request sent to your customers." +msgstr "" +"Stelle hier ein, wie viele Tage zwischen der Bestellung und dem Versand der " +"Bewertungserinnerung liegen soll." + +#: includes/class-wc-trusted-shops-admin.php:903 +msgctxt "trusted-shops" +msgid "Start export" +msgstr "Export starten" + +#: includes/class-wc-trusted-shops-core.php:55 +#: includes/class-wc-trusted-shops-core.php:64 +#: includes/class-wc-ts-dependencies.php:36 +#: includes/class-wc-ts-dependencies.php:45 +msgctxt "trusted-shops" +msgid "Cheatin’ huh?" +msgstr "So geht das leider nicht.." + +#: includes/class-wc-trusted-shops-core.php:213 +msgctxt "trusted-shops" +msgid "Yes" +msgstr "Ja" + +#: includes/class-wc-trusted-shops-core.php:213 +msgctxt "trusted-shops" +msgid "No" +msgstr "Nein" + +#: includes/class-wc-trusted-shops-core.php:261 +#, php-format +msgctxt "trusted-shops" +msgid "" +"If the App helped you, please leave a %s★★" +"★★★%s in the Wordpress plugin repository." +msgstr "" +"Wenn dir die App hilft, hinterlasse bitte eine %s★" +"★★★★%s Bewertung in der WordPress Plugin Bibliothek." + +#: includes/class-wc-trusted-shops-core.php:410 +msgctxt "trusted-shops" +msgid "Settings" +msgstr "Einstellungen" + +#: includes/class-wc-trusted-shops-review-exporter.php:64 +msgctxt "trusted-shops" +msgid "Order ID" +msgstr "Bestellnummer" + +#: includes/class-wc-trusted-shops-review-exporter.php:65 +msgctxt "trusted-shops" +msgid "Order date" +msgstr "Bestelldatum" + +#: includes/class-wc-trusted-shops-review-exporter.php:66 +msgctxt "trusted-shops" +msgid "# Days" +msgstr "# Tage" + +#: includes/class-wc-trusted-shops-review-exporter.php:67 +msgctxt "trusted-shops" +msgid "Email" +msgstr "E-Mail" + +#: includes/class-wc-trusted-shops-review-exporter.php:68 +msgctxt "trusted-shops" +msgid "First name" +msgstr "Vorname" + +#: includes/class-wc-trusted-shops-review-exporter.php:69 +msgctxt "trusted-shops" +msgid "Last name" +msgstr "Nachname" + +#: includes/class-wc-trusted-shops-template-hooks.php:121 +#, php-format +msgctxt "trusted-shops" +msgid "" +"Your review reminder e-mail has been cancelled successfully. Return to %s." +msgstr "" +"Deine Bewertungserinnerung wurde erfolgreich deaktiviert. Kehre zurück zur " +"%s." + +#: includes/class-wc-trusted-shops-template-hooks.php:121 +msgctxt "trusted-shops" +msgid "Home" +msgstr "Startseite" + +#: includes/class-wc-trusted-shops-template-hooks.php:193 +msgctxt "trusted-shops" +msgid "" +"Yes, I would like to be reminded via e-mail after {days} day(s) to review my " +"order. I am able to cancel the reminder at any time by clicking on the " +"\"cancel review reminder\" link within the order confirmation." +msgstr "" +"Ja, ich bin damit einverstanden, eine Erinnerung per E-Mail zur Bewertung " +"nach {days} Tage(n) zu erhalten. Ich kann mich jederzeit davon abmelden bzw. " +"widersprechen, indem ich dem Link „von der Bewertungserinnerung abmelden“ in " +"der Bestellbestätigung folge." + +#: includes/class-wc-trusted-shops-template-hooks.php:198 +msgctxt "trusted-shops" +msgid "Please allow us to send a review reminder by e-mail." +msgstr "Bitte akzeptiere den Erhalt einer Bewertungserinnerung per E-Mail." + +#: includes/class-wc-trusted-shops-template-hooks.php:201 +msgctxt "trusted-shops" +msgid "Review reminder" +msgstr "Bewertungs Erinnerung" + +#: includes/class-wc-trusted-shops-template-hooks.php:202 +msgctxt "trusted-shops" +msgid "Asks the customer to receive a Trusted Shops review reminder." +msgstr "" +"Holt die Erlaubnis zum Senden einer einmaligen Trusted Shops " +"Bewertungserinnerung ein." + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:24 +msgctxt "trusted-shops" +msgid "Trusted Shops Review Reminder" +msgstr "Trusted Shops Bewertungs-Erinnerung" + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:25 +msgctxt "trusted-shops" +msgid "" +"This E-Mail is being sent to a customer to remind him about the possibility " +"to leave a review at Trusted Shops." +msgstr "" +"Diese E-Mail wird einmalig an Kunden verschickt, um diese an die Abgabe " +"einer Bewertung bei Trusted Shops zu erinnern." + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:54 +msgctxt "trusted-shops" +msgid "Please rate your {site_title} order from {order_date}" +msgstr "Bitte bewerte deinen Einkauf vom {order_date} bei {site_title}" + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:64 +msgctxt "trusted-shops" +msgid "Please rate your Order" +msgstr "Bitte bewerte deinen Einkauf" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:19 +msgctxt "trusted-shops" +msgid "Show your TS shop review sticker." +msgstr "Zeige deinen Trusted Shops Review Sticker an." + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:21 +msgctxt "trusted-shops" +msgid "Trusted Shops Shop Review Sticker" +msgstr "Trusted Shops Shop Review Sticker" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:25 +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:46 +msgctxt "trusted-shops" +msgid "Trusted Shops Reviews" +msgstr "Trusted Shops Bewertung" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:26 +msgctxt "trusted-shops" +msgid "Title" +msgstr "Bezeichnung" + +#: src/Package.php:55 +msgctxt "trusted-shops" +msgid "" +"Trustbadge Reviews for WooCommerce needs at least WooCommerce version 3.1 to " +"run." +msgstr "" +"Trustbadge Reviews for WooCommerce benötigt mind. WooCommerce Version 3.1 um " +"zu starten." + +#: templates/emails/customer-trusted-shops.php:17 +#: templates/emails/plain/customer-trusted-shops.php:14 +#, php-format +msgctxt "trusted-shops" +msgid "Dear %s %s," +msgstr "Hallo %s %s," + +#: templates/emails/customer-trusted-shops.php:18 +#: templates/emails/plain/customer-trusted-shops.php:16 +#, php-format +msgctxt "trusted-shops" +msgid "" +"You have recently shopped at %s. Thank you! We would be glad if you spent " +"some time to write a review about your order. To do so please follow follow " +"the link." +msgstr "" +"Du hast vor einiger Zeit bei %s eingekauft. Vielen Dank! Wir wären froh " +"darüber, wenn du dir die Zeit nimmst und deinen Einkauf bewerten würdest. Um " +"dies nun zu tun, klicke bitte auf den nachfolgenden Link." + +#: templates/emails/customer-trusted-shops.php:22 +msgctxt "trusted-shops" +msgid "Rate Order now" +msgstr "Einkauf jetzt bewerten" + +#~ msgctxt "trusted-shops" +#~ msgid "Duplicate Plugin installation" +#~ msgstr "Doppelte Plugin Installation" + +#, php-format +#~ msgctxt "trusted-shops" +#~ msgid "" +#~ "It seems like you've installed WooCommerce Germanized and Trustbadge " +#~ "Reviews for WooCommerce. Please deactivate Trustbadge Reviews for " +#~ "WooCommerce as long as you are using WooCommerce Germanized. You can " +#~ "manage your Trusted Shops configuration within your %s." +#~ msgstr "" +#~ "Es scheint als hättest du WooCommerce Germanized und Trustbadge Reviews " +#~ "for WooCommerce installiert. Bitte deaktiviere das Trustbadge Reviews " +#~ "Plugin solange du WooCommerce Germanized verwendest. Du kannst deine " +#~ "Trusted Shops Konfiguration in deinen %s verwalten." + +#~ msgctxt "trusted-shops" +#~ msgid "Germanized settings" +#~ msgstr "Germanized Einstellungen" + +#~ msgctxt "trusted-shops" +#~ msgid "Deactivate standalone version" +#~ msgstr "Deaktiviere die Standalone-Version" + +#~ msgctxt "trusted-shops" +#~ msgid "(WooCommerce Product Reviews will be replaced)" +#~ msgstr "(WooCommerce Produktbewertungen werden ersetzt)" + +#, php-format +#~ msgid "" +#~ "Please install WooCommerce before " +#~ "installing WooCommerce Germanized. Thank you!" +#~ msgstr "" +#~ "Bitte installiere WooCommerce bevor " +#~ "du WooCommerce Germanized installierst. Vielen Dank!" diff --git a/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE_formal.mo b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE_formal.mo new file mode 100644 index 000000000..4ea14e41a Binary files /dev/null and b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE_formal.mo differ diff --git a/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE_formal.po b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE_formal.po new file mode 100644 index 000000000..1b18aea99 --- /dev/null +++ b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-de_DE_formal.po @@ -0,0 +1,1172 @@ +msgid "" +msgstr "" +"Project-Id-Version: WooCommerce Trusted Shops\n" +"POT-Creation-Date: 2019-10-15 12:40+0200\n" +"PO-Revision-Date: 2019-10-15 12:41+0200\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: de\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"X-Generator: Poedit 2.2.4\n" +"X-Poedit-Basepath: ../..\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Poedit-KeywordsList: __;_e;__ngettext:1,2;_n:1,2;__ngettext_noop:1,2;" +"_n_noop:1,2;_c,_nc:4c,1,2;_x:1,2c;_ex:1,2c;_nx:4c,1,2;_nx_noop:4c,1,2;" +"esc_attr_e;esc_html_e;esc_html__\n" +"X-Poedit-SearchPath-0: .\n" +"X-Poedit-SearchPathExcluded-0: node_modules\n" +"X-Poedit-SearchPathExcluded-1: vendor\n" + +#: includes/admin/settings/class-wc-ts-gzd-settings-tab.php:48 +msgctxt "trusted-shops" +msgid "Setup your Trusted Shops Integration." +msgstr "Setzen Sie Ihre Trusted Shops Integration auf." + +#: includes/admin/settings/class-wc-ts-gzd-settings-tab.php:52 +#: includes/class-wc-ts-settings-handler.php:23 +msgctxt "trusted-shops" +msgid "Trusted Shops" +msgstr "Trusted Shops" + +#: includes/admin/views/html-notice-dependencies.php:16 +msgctxt "trusted-shops" +msgid "Dependencies Missing or Outdated" +msgstr "Wichtige Plugins fehlen oder sind veraltet" + +#: includes/admin/views/html-notice-dependencies.php:24 +msgctxt "trusted-shops" +msgid "" +"To use WooCommerce Trusted Shops you may at first install the following " +"plugins:" +msgstr "" +"Um WooCommerce Trusted Shops zuverlässig nutzen zu können, müssen Sie zuerst " +"folgende Plugins installieren:" + +#: includes/admin/views/html-notice-dependencies.php:28 +#, php-format +msgctxt "trusted-shops" +msgid "Install %s" +msgstr "Installiere %s" + +#: includes/admin/views/html-notice-dependencies.php:36 +msgctxt "trusted-shops" +msgid "" +"To use WooCommerce Trusted Shops you may at first update the following " +"plugins to a newer version:" +msgstr "" +"Um WooCommerce Trusted Shops zuverlässig nutzen zu können, updaten Sie bitte " +"folgende Plugins:" + +#: includes/admin/views/html-notice-dependencies.php:40 +#, php-format +msgctxt "trusted-shops" +msgid "%s required in at least version %s" +msgstr "%s wird mindestens in Version %s benötigt" + +#: includes/admin/views/html-notice-dependencies.php:49 +msgctxt "trusted-shops" +msgid "Check for Updates" +msgstr "nach Updates suchen" + +#: includes/admin/views/html-notice-dependencies.php:50 +msgctxt "trusted-shops" +msgid "or" +msgstr "oder" + +#: includes/admin/views/html-notice-dependencies.php:51 +msgctxt "trusted-shops" +msgid "Install an older version" +msgstr "Installiere eine ältere Version" + +#: includes/admin/views/html-notice-update.php:14 +msgctxt "trusted-shops" +msgid "" +"WooCommerce Trusted Shops Data Update Required – We " +"just need to update your installation to the latest version" +msgstr "" +"WooCommerce Trusted Shops Datenaktualisierung erforderlich " +"– Wir müssen deine Installation auf die neueste Version updaten" + +#: includes/admin/views/html-notice-update.php:15 +msgctxt "trusted-shops" +msgid "Run the updater" +msgstr "Update starten" + +#: includes/admin/views/html-notice-update.php:19 +msgctxt "trusted-shops" +msgid "" +"It is strongly recommended that you backup your database before proceeding. " +"Are you sure you wish to run the updater now?" +msgstr "" +"Sie sollten vor einem Update immer ein Backup der Datenbank anlegen. Sind " +"Sie sicher das Update jetzt zu installieren?" + +#: includes/admin/views/html-wpml-notice.php:11 +msgctxt "trusted-shops" +msgid "WPML Support" +msgstr "WPML Unterstützung" + +#: includes/admin/views/html-wpml-notice.php:14 +msgctxt "trusted-shops" +msgid "" +"These settings serve as default settings for all your languages. To adjust " +"the settings for a certain language, please switch your admin language " +"through the WPML language switcher and adjust the corresponding settings." +msgstr "" +"Diese Einstellungen werden als Standard für alle Ihre Sprachen verwendet. Um " +"die Einstellungen für eine spezielle Sprache zu ändern, wechseln Sie bitte " +"zur entsprechenden Sprache über den WPML Sprachumschalter." + +#: includes/admin/views/html-wpml-notice.php:16 +#, php-format +msgctxt "trusted-shops" +msgid "" +"These settings apply for your %s shop. To adjust settings for another " +"language, please switch your admin language through the WPML language " +"switcher." +msgstr "" +"Diese Einstellungen werden für Ihren %s Shop verwendet. Um die Einstellungen " +"für eine andere Sprache anzupassen, wechseln Sie bitte die Sprache über den " +"WPML Sprachumschalter." + +#: includes/class-wc-trusted-shops-admin.php:84 +#: includes/class-wc-trusted-shops-admin.php:121 +msgctxt "trusted-shops" +msgid "GTIN" +msgstr "GTIN" + +#: includes/class-wc-trusted-shops-admin.php:84 +#: includes/class-wc-trusted-shops-admin.php:121 +msgctxt "trusted-shops" +msgid "" +"ID that allows your products to be identified worldwide. If you want to " +"display your Trusted Shops Product Reviews in Google Shopping and paid " +"Google adverts, Google needs the GTIN." +msgstr "" +"Identifikationsnummer, mit der Produkte weltweit eindeutig identifiziert " +"werden können. Wenn Sie Ihre Trusted Shops Produktbewertungen in Google " +"Shopping und bezahlten Google Produktanzeigen ausspielen möchten, benötigt " +"Google die GTIN." + +#: includes/class-wc-trusted-shops-admin.php:88 +#: includes/class-wc-trusted-shops-admin.php:122 +msgctxt "trusted-shops" +msgid "MPN" +msgstr "MPN" + +#: includes/class-wc-trusted-shops-admin.php:88 +#: includes/class-wc-trusted-shops-admin.php:122 +msgctxt "trusted-shops" +msgid "" +"If you don't have a GTIN for your products, you can pass the brand name and " +"the MPN on to Google to use the Trusted Shops Google Integration." +msgstr "" +"Wenn Ihre Produkte keine GTIN haben, können SIe den Markennamen zusammen mit " +"der MPN übergeben, um die Trusted Shops Google Integration zu nutzen." + +#: includes/class-wc-trusted-shops-admin.php:142 +msgctxt "trusted-shops" +msgid "Brand" +msgstr "Marke" + +#: includes/class-wc-trusted-shops-admin.php:176 +msgctxt "trusted-shops" +msgid "This field is mandatory" +msgstr "Dieses Feld ist ein Pflichtfeld" + +#: includes/class-wc-trusted-shops-admin.php:183 +msgctxt "trusted-shops" +msgid "Trusted Shops Options" +msgstr "Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:190 +msgctxt "trusted-shops" +msgid "Arial" +msgstr "Arial" + +#: includes/class-wc-trusted-shops-admin.php:191 +msgctxt "trusted-shops" +msgid "Geneva" +msgstr "Geneva" + +#: includes/class-wc-trusted-shops-admin.php:192 +msgctxt "trusted-shops" +msgid "Georgia" +msgstr "Georgia" + +#: includes/class-wc-trusted-shops-admin.php:193 +msgctxt "trusted-shops" +msgid "Helvetica" +msgstr "Helvetica" + +#: includes/class-wc-trusted-shops-admin.php:194 +msgctxt "trusted-shops" +msgid "Sans-serif" +msgstr "Sans-serif" + +#: includes/class-wc-trusted-shops-admin.php:195 +msgctxt "trusted-shops" +msgid "Serif" +msgstr "Serif" + +#: includes/class-wc-trusted-shops-admin.php:196 +msgctxt "trusted-shops" +msgid "Trebuchet MS" +msgstr "Trebuchet MS" + +#: includes/class-wc-trusted-shops-admin.php:197 +msgctxt "trusted-shops" +msgid "Verdana" +msgstr "Verdana" + +#: includes/class-wc-trusted-shops-admin.php:220 +msgctxt "trusted-shops" +msgid "Trusted Shops Integration" +msgstr "Trusted Shops Integration" + +#: includes/class-wc-trusted-shops-admin.php:221 +#, php-format +msgctxt "trusted-shops" +msgid "Do you need help with integrating your Trustbadge? %s" +msgstr "Brauchen Sie Hilfe bei der Einbindung Ihres Trustbadges? %s" + +#: includes/class-wc-trusted-shops-admin.php:221 +msgctxt "trusted-shops" +msgid "To the step-by-step instructions" +msgstr "Zur Schritt-für-Schritt Anleitung" + +#: includes/class-wc-trusted-shops-admin.php:227 +msgctxt "trusted-shops" +msgid "Trusted Shops ID" +msgstr "Trusted Shops ID" + +#: includes/class-wc-trusted-shops-admin.php:228 +msgctxt "trusted-shops" +msgid "" +"The Trusted Shops ID is a unique identifier for your shop. You can find your " +"Trusted Shops ID in your My Trusted Shops account." +msgstr "" +"Die Trusted Shops ID identifiziert Ihren Shop eindeutig bei Trusted Shops. " +"Sie finden die Trusted Shops ID in Ihrem My Trusted Shops Account." + +#: includes/class-wc-trusted-shops-admin.php:237 +msgctxt "trusted-shops" +msgid "Edit Mode" +msgstr "Bearbeitungsmodus" + +#: includes/class-wc-trusted-shops-admin.php:239 +#: includes/class-wc-trusted-shops-admin.php:300 +#: includes/class-wc-trusted-shops-admin.php:423 +msgctxt "trusted-shops" +msgid "" +"The advanced configuration is for users with programming skills. Here you " +"can create even more individual settings." +msgstr "" +"Der Expertenmodus ist für fortgeschrittene Nutzer mit Programmierkenntnissen " +"gedacht. Hier können Sie noch mehr individuelle Einstellungen vornehmen." + +#: includes/class-wc-trusted-shops-admin.php:243 +msgctxt "trusted-shops" +msgid "Standard configuration" +msgstr "Standardmodus" + +#: includes/class-wc-trusted-shops-admin.php:244 +msgctxt "trusted-shops" +msgid "Advanced configuration" +msgstr "Expertenmodus" + +#: includes/class-wc-trusted-shops-admin.php:252 +msgctxt "trusted-shops" +msgid "Configure your Trustbadge" +msgstr "Konfigurieren Sie Ihr Trustbadge" + +#: includes/class-wc-trusted-shops-admin.php:258 +msgctxt "trusted-shops" +msgid "Display Trustbadge" +msgstr "Trustbadge anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:260 +msgctxt "trusted-shops" +msgid "Display the Trustbadge on all the pages of your shop." +msgstr "Zeige das Trustbadge auf allen Seiten des Shops an." + +#: includes/class-wc-trusted-shops-admin.php:267 +msgctxt "trusted-shops" +msgid "Variant" +msgstr "Variante" + +#: includes/class-wc-trusted-shops-admin.php:269 +msgctxt "trusted-shops" +msgid "You can display your Trustbadge with or without Review Stars." +msgstr "Sie können Ihr Trustbadge mit oder ohne Bewertungssterne anzeigen." + +#: includes/class-wc-trusted-shops-admin.php:273 +#: includes/class-wc-trusted-shops-admin.php:763 +msgctxt "trusted-shops" +msgid "Display Trustbadge with review stars" +msgstr "Trustbadge mit Bewertungssternen anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:274 +#: includes/class-wc-trusted-shops-admin.php:767 +msgctxt "trusted-shops" +msgid "Display Trustbadge without review stars" +msgstr "Trustbadge ohne Bewertungssterne anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:280 +msgctxt "trusted-shops" +msgid "Vertical Offset" +msgstr "Vertikaler Abstand" + +#: includes/class-wc-trusted-shops-admin.php:281 +msgctxt "trusted-shops" +msgid "" +"Choose the distance that the Trustbadge will appear from the bottom-right " +"corner of the screen." +msgstr "" +"Wählen Sie einen Abstand für Ihr Trustbadge vom unteren rechten " +"Bildschirmrand." + +#: includes/class-wc-trusted-shops-admin.php:284 +#: includes/class-wc-trusted-shops-admin.php:492 +#: includes/class-wc-trusted-shops-admin.php:543 +#: includes/class-wc-trusted-shops-admin.php:558 +msgctxt "trusted-shops" +msgid "px" +msgstr "px" + +#: includes/class-wc-trusted-shops-admin.php:290 +#: includes/class-wc-trusted-shops-admin.php:499 +#: includes/class-wc-trusted-shops-admin.php:549 +#: includes/class-wc-trusted-shops-admin.php:565 +#, php-format +msgctxt "trusted-shops" +msgid "Please choose a non-negative number (at least %d)" +msgstr "Bitte verwenden Sie eine nicht-negative Nummer (mindestens %d)" + +#: includes/class-wc-trusted-shops-admin.php:296 +msgctxt "trusted-shops" +msgid "Trustbadge code" +msgstr "Trustbadge Code" + +#: includes/class-wc-trusted-shops-admin.php:308 +msgctxt "trusted-shops" +msgid "Configure your Shop Reviews" +msgstr "Konfigurieren Sie Ihre Shopbewertungen" + +#: includes/class-wc-trusted-shops-admin.php:314 +msgctxt "trusted-shops" +msgid "Display Shop Review Sticker" +msgstr "Shopbewertungssticker anzeigen" + +#: includes/class-wc-trusted-shops-admin.php:315 +msgctxt "trusted-shops" +msgid "" +"To display the Shop Review Sticker, you have to assign the widget \"Trusted " +"Shops Review Sticker\"." +msgstr "" +"Um den Shopbewertungssticker anzuzeigen, müssen Sie das Widget „Trusted " +"Shops Shopbewertungssticker“ zuweisen." + +#: includes/class-wc-trusted-shops-admin.php:316 +#, php-format +msgctxt "trusted-shops" +msgid "Assign widget %s" +msgstr "Widget %s zuweisen" + +#: includes/class-wc-trusted-shops-admin.php:316 +#: includes/class-wc-trusted-shops-admin.php:588 +#: includes/class-wc-trusted-shops-admin.php:896 +msgctxt "trusted-shops" +msgid "here" +msgstr "hier" + +#: includes/class-wc-trusted-shops-admin.php:324 +#: includes/class-wc-trusted-shops-admin.php:472 +msgctxt "trusted-shops" +msgid "Background color" +msgstr "Hintergrundfarbe" + +#: includes/class-wc-trusted-shops-admin.php:325 +msgctxt "trusted-shops" +msgid "Choose the background color for your Review Sticker." +msgstr "Wählen Sie die Hintergrundfarbe für den Review Sticker." + +#: includes/class-wc-trusted-shops-admin.php:332 +msgctxt "trusted-shops" +msgid "Font" +msgstr "Schriftart" + +#: includes/class-wc-trusted-shops-admin.php:334 +msgctxt "trusted-shops" +msgid "Choose the font for your Review Sticker." +msgstr "Wählen Sie die Schriftart für den Review Sticker." + +#: includes/class-wc-trusted-shops-admin.php:341 +msgctxt "trusted-shops" +msgid "Number of reviews displayed" +msgstr "Anzahl Bewertungen" + +#: includes/class-wc-trusted-shops-admin.php:342 +msgctxt "trusted-shops" +msgid "" +"Display x alternating Shop Reviews in your Shop Review Sticker. You can " +"display between 1 and 5 alternating Shop Reviews." +msgstr "" +"Zeige x Shopbewertungen im Wechsel in Ihrem Shopbewertungssticker an. Es " +"können mindestens 1 und maximal 5 Bewertungen im Wechsel angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:345 +msgctxt "trusted-shops" +msgid "Show x alternating reviews" +msgstr "x Bewertungen im Wechsel zeigen" + +#: includes/class-wc-trusted-shops-admin.php:352 +#: includes/class-wc-trusted-shops-admin.php:369 +#, php-format +msgctxt "trusted-shops" +msgid "Please choose a non-negative number between %d and %d" +msgstr "Bitte verwenden Sie eine nicht-negative Nummer zwischen %d und %d" + +#: includes/class-wc-trusted-shops-admin.php:358 +msgctxt "trusted-shops" +msgid "Minimum rating displayed" +msgstr "Angezeigte Mindestnote" + +#: includes/class-wc-trusted-shops-admin.php:359 +msgctxt "trusted-shops" +msgid "Only show Shop Reviews with a minimum rating of x stars. " +msgstr "Zeige nur Shopbewertungen mit einer Mindestanzahl von x Sternen." + +#: includes/class-wc-trusted-shops-admin.php:362 +msgctxt "trusted-shops" +msgid "Star(s)" +msgstr "Stern(e)" + +#: includes/class-wc-trusted-shops-admin.php:375 +msgctxt "trusted-shops" +msgid "Sticker code" +msgstr "Sticker Code" + +#: includes/class-wc-trusted-shops-admin.php:379 +#: includes/class-wc-trusted-shops-admin.php:506 +#: includes/class-wc-trusted-shops-admin.php:571 +msgctxt "trusted-shops" +msgid "" +"The advanced configuration is for users with programming skills. Here you " +"can perform even more individual settings." +msgstr "" +"Der Expertenmodus ist für fortgeschrittene Nutzer mit Programmierkenntnissen " +"gedacht. Hier können Sie noch mehr individuelle Einstellungen vornehmen." + +#: includes/class-wc-trusted-shops-admin.php:385 +msgctxt "trusted-shops" +msgid "Google Organic Search" +msgstr "Organische Google Suche" + +#: includes/class-wc-trusted-shops-admin.php:386 +msgctxt "trusted-shops" +msgid "" +"Activate this option to give Google the opportunity to show your Shop " +"Reviews in Google organic search results." +msgstr "" +"Aktivieren Sie diese Funktion, um Google die Möglichkeit zu geben, Ihre " +"Shopbewertungen in den organischen Google Suchergebnissen anzuzeigen." + +#: includes/class-wc-trusted-shops-admin.php:387 +msgctxt "trusted-shops" +msgid "" +"By activating this option, rich snippets will be integrated in the selected " +"pages so your shop review stars may be displayed in Google organic search " +"results. If you use Product Reviews and already activated rich snippets in " +"expert mode, we recommend integrating rich snippets for Shop Reviews on " +"category pages only." +msgstr "" +"Wenn Sie diese Option aktivieren, werden Rich Snippets in die ausgewählten " +"Seiten eingebunden, sodass Ihre Shopbewertungssterne in den organischen " +"Google Suchergebnissen angezeigt werden können. Wenn Sie Produktbewertungen " +"aktiviert haben und dort im Expertenmodus bereits Rich Snippets aktiviert " +"haben, empfehlen wir die Ausgabe der Rich Snippets für Shopbewertungen nur " +"auf der Kategorieseite." + +#: includes/class-wc-trusted-shops-admin.php:394 +msgctxt "trusted-shops" +msgid "Activate rich snippets on" +msgstr "Rich Snippets aktivieren auf" + +#: includes/class-wc-trusted-shops-admin.php:395 +msgctxt "trusted-shops" +msgid "category pages" +msgstr "Kategorieseiten" + +#: includes/class-wc-trusted-shops-admin.php:403 +msgctxt "trusted-shops" +msgid "product pages" +msgstr "Produktdetailseiten" + +#: includes/class-wc-trusted-shops-admin.php:411 +msgctxt "trusted-shops" +msgid "homepage (not recommended)" +msgstr "Startseite (nicht empfohlen)" + +#: includes/class-wc-trusted-shops-admin.php:419 +msgctxt "trusted-shops" +msgid "Rich snippets code" +msgstr "Rich Snippets Code" + +#: includes/class-wc-trusted-shops-admin.php:431 +msgctxt "trusted-shops" +msgid "Configure your Product Reviews " +msgstr "Konfigurieren Sie die Produktbewertungen" + +#: includes/class-wc-trusted-shops-admin.php:432 +#, php-format +msgctxt "trusted-shops" +msgid "To use Product Reviews, activate them in your %s first." +msgstr "" +"Um Produktbewertungen nutzen zu können, schalten Sie diese zuerst in Ihrem " +"%s frei." + +#: includes/class-wc-trusted-shops-admin.php:432 +msgctxt "trusted-shops" +msgid "Trusted Shops package" +msgstr "Trusted Shops Paket" + +#: includes/class-wc-trusted-shops-admin.php:438 +msgctxt "trusted-shops" +msgid "Collect Product Reviews" +msgstr "Produktbewertungen sammeln" + +#: includes/class-wc-trusted-shops-admin.php:439 +msgctxt "trusted-shops" +msgid "" +"Show Product Reviews on the product page in a separate tab, just as shown on " +"the picture on the right." +msgstr "" +"Zeige Produktbewertungen auf der Produktseite in einem separaten Reiter, wie " +"in der Grafik rechts abgebildet." + +#: includes/class-wc-trusted-shops-admin.php:447 +msgctxt "trusted-shops" +msgid "Reviews" +msgstr "Bewertungen" + +#: includes/class-wc-trusted-shops-admin.php:448 +#: includes/class-wc-trusted-shops-admin.php:457 +msgctxt "trusted-shops" +msgid "You can choose a name for the tab with your Product Reviews." +msgstr "" +"Sie können selbst bestimmen, wie der Reiter in dem Ihre Produktbewertungen " +"angezeigt werden, heißen soll." + +#: includes/class-wc-trusted-shops-admin.php:449 +msgctxt "trusted-shops" +msgid "Show Product Reviews on the product detail page in an additional tab." +msgstr "Produktbewertungen auf der Produktseite in separatem Reiter anzeigen." + +#: includes/class-wc-trusted-shops-admin.php:456 +msgctxt "trusted-shops" +msgid "Name of Product Reviews tab" +msgstr "Bezeichnung Reiter" + +#: includes/class-wc-trusted-shops-admin.php:460 +msgctxt "trusted-shops" +msgid "Product reviews" +msgstr "Produktbewertungen" + +#: includes/class-wc-trusted-shops-admin.php:464 +msgctxt "trusted-shops" +msgid "Border color" +msgstr "Umrandung" + +#: includes/class-wc-trusted-shops-admin.php:465 +msgctxt "trusted-shops" +msgid "Set the color for the frame around your Product Reviews." +msgstr "Legen Sie die Farbe für den Rahmen um die Produktbewertungen fest." + +#: includes/class-wc-trusted-shops-admin.php:473 +msgctxt "trusted-shops" +msgid "Set the background color for your Product Reviews." +msgstr "Legen Sie die Hintergrundfarbe für die Produktbewertungen fest." + +#: includes/class-wc-trusted-shops-admin.php:480 +#: includes/class-wc-trusted-shops-admin.php:530 +msgctxt "trusted-shops" +msgid "Star color" +msgstr "Farbe der Sterne" + +#: includes/class-wc-trusted-shops-admin.php:481 +msgctxt "trusted-shops" +msgid "Set the color for the Product Review stars in your Product Reviews tab." +msgstr "" +"Legen Sie die Farbe der Sterne fest, die in Ihrem Produktbewertungs-Reiter " +"angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:488 +#: includes/class-wc-trusted-shops-admin.php:538 +msgctxt "trusted-shops" +msgid "Star size" +msgstr "Größe der Sterne" + +#: includes/class-wc-trusted-shops-admin.php:493 +msgctxt "trusted-shops" +msgid "Set the size for the Product Review stars in your Product Reviews tab." +msgstr "" +"Legen Sie die Größe der Sterne fest, die in Ihrem Produktbewertungs-Reiter " +"angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:504 +msgctxt "trusted-shops" +msgid "Product Sticker Code" +msgstr "Produktbewertungen Code" + +#: includes/class-wc-trusted-shops-admin.php:513 +#: includes/class-wc-trusted-shops-admin.php:579 +msgctxt "trusted-shops" +msgid "jQuerySelector" +msgstr "jQuerySelector" + +#: includes/class-wc-trusted-shops-admin.php:514 +msgctxt "trusted-shops" +msgid "" +"Please choose where your Product Reviews shall be displayed on the Product " +"detail page." +msgstr "" +"Wählen Sie, wo die Produktbewertungen auf der Produktdetailseite angezeigt " +"werden sollen." + +#: includes/class-wc-trusted-shops-admin.php:521 +msgctxt "trusted-shops" +msgid "Rating stars" +msgstr "Bewertungssterne" + +#: includes/class-wc-trusted-shops-admin.php:522 +msgctxt "trusted-shops" +msgid "Show star ratings on the product detail page below your product name." +msgstr "Bewertungssterne auf der Produktseite unter dem Produktnamen anzeigen." + +#: includes/class-wc-trusted-shops-admin.php:523 +msgctxt "trusted-shops" +msgid "" +"Display Product Review stars on product pages below the product name, just " +"as shown in the picture on the right." +msgstr "" +"Zeige Bewertungssterne auf der Produktseite unter dem Produktnamen, wie in " +"der Grafik rechts abgebildet." + +#: includes/class-wc-trusted-shops-admin.php:532 +msgctxt "trusted-shops" +msgid "" +"Set the color for the review stars, that are displayed on the product page, " +"below your product name." +msgstr "" +"Legen Sie die Farbe der Sterne fest, die auf der Produktseite unter dem " +"Produktnamen angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:540 +msgctxt "trusted-shops" +msgid "" +"Set the size for the review stars that are displayed on the product page, " +"below your product name." +msgstr "" +"Legen Sie die Größe der Sterne fest, die auf der Produktseite unter dem " +"Produktnamen angezeigt werden." + +#: includes/class-wc-trusted-shops-admin.php:554 +msgctxt "trusted-shops" +msgid "Font size" +msgstr "Schriftgröße" + +#: includes/class-wc-trusted-shops-admin.php:556 +msgctxt "trusted-shops" +msgid "Set the font size for the text that goes with your review stars." +msgstr "Legen Sie die Schriftgröße für den Text zu den Bewertungssternen fest." + +#: includes/class-wc-trusted-shops-admin.php:570 +msgctxt "trusted-shops" +msgid "Product Review Code" +msgstr "Produktbewertungen Code" + +#: includes/class-wc-trusted-shops-admin.php:580 +msgctxt "trusted-shops" +msgid "" +"Please choose where your Product Review Stars shall be displayed on the " +"Product Detail page." +msgstr "" +"Wählen Sie, wo die Produktbewertungssterne auf der Produktdetailseite " +"angezeigt werden sollen." + +#: includes/class-wc-trusted-shops-admin.php:587 +msgctxt "trusted-shops" +msgid "Brand attribute" +msgstr "Marken-Produktattribut" + +#: includes/class-wc-trusted-shops-admin.php:588 +#, php-format +msgctxt "trusted-shops" +msgid "Create brand attribute %s" +msgstr "Marken-Produktattribut %s erstellen" + +#: includes/class-wc-trusted-shops-admin.php:589 +msgctxt "trusted-shops" +msgid "" +"Brand name of the product. By passing this information on to Google, you " +"improve your chances of having Google identify your products. Assign your " +"brand attribute. If your products don't have a GTIN, you can pass on the " +"brand name and the MPN to use Google Integration." +msgstr "" +"Markenname des Produkts. Durch Übergeben dieser Variable verbessern Sie die " +"Möglichkeit Ihre Produkte von Google eindeutig identifizieren zu lassen. " +"Weisen Sie ein Marken-Produktattribut zu. Wenn Ihre Produkte keine GTIN " +"haben, können Sie den Markennamen zusammen mit der MPN übergeben, um die " +"Google Integration zu nutzen." + +#: includes/class-wc-trusted-shops-admin.php:595 +msgctxt "trusted-shops" +msgid "None" +msgstr "Keine" + +#: includes/class-wc-trusted-shops-admin.php:606 +msgctxt "trusted-shops" +msgid "Configure your Review Requests" +msgstr "Konfigurieren Sie die Bewertungserinnerungen" + +#: includes/class-wc-trusted-shops-admin.php:607 +msgctxt "trusted-shops" +msgid "" +"7 days after an order has been placed, Trusted Shops automatically sends an " +"invite to your customers. If you want to set a different time for sending " +"automatic Review Requests, please activate the option below. If you want to " +"send review requests with legal certainty, you need your customers' consent " +"to receive Review Requests. You also have to include an option to " +"unsubscribe." +msgstr "" +"Trusted Shops versendet 7 Tage nach Bestellung automatisch eine " +"Bewertungserinnerung an Ihre Kunden. Wenn Sie lieber selbst entscheiden " +"möchten, wann Bewertungsanfragen versendet werden, aktivieren Sie die " +"untenstehende Option. Möchten Sie rechtssicher Bewertungs-Erinnerungen " +"verschicken, so benötigen Sie die ausdrückliche Einwilligung Ihrer Kunden " +"zum Empfang von Bewertungsemails. Sie müssen Ihren Kunden zudem eine " +"Möglichkeit geben, Bewertungserinnerungen wieder abzubestellen." + +#: includes/class-wc-trusted-shops-admin.php:613 +msgctxt "trusted-shops" +msgid "Enable Review Requests" +msgstr "Bewertungserinnerungen aktivieren" + +#: includes/class-wc-trusted-shops-admin.php:622 +msgctxt "trusted-shops" +msgid "WooCommerce status" +msgstr "Bestellstatus" + +#: includes/class-wc-trusted-shops-admin.php:623 +msgctxt "trusted-shops" +msgid "" +"We recommend choosing the order status that you set when your products have " +"been shipped." +msgstr "" +"Wählen Sie hier am besten den Bestellstatus aus WooCommerce aus, den Sie " +"einstellen, wenn Ihre Ware versendet wurde." + +#: includes/class-wc-trusted-shops-admin.php:631 +msgctxt "trusted-shops" +msgid "Days until Review Request" +msgstr "Tage bis zur Erinnerung" + +#: includes/class-wc-trusted-shops-admin.php:632 +msgctxt "trusted-shops" +msgid "" +"Set the number of days to wait after an order has reached the order status " +"you selected above before having a review request sent to your customers." +msgstr "" +"Stellen Sie hier ein nach wie vielen Tagen nach Erreichen des oben " +"ausgewählten Bestellstatus die Bewertungserinnerung an den Käufer versendet " +"werden soll." + +#: includes/class-wc-trusted-shops-admin.php:644 +msgctxt "trusted-shops" +msgid "Permission via checkbox" +msgstr "Einwilligung per Checkbox" + +#: includes/class-wc-trusted-shops-admin.php:645 +msgctxt "trusted-shops" +msgid "" +"If the checkbox is activated, only customers who gave their consent will " +"receive Review Requests." +msgstr "" +"Bei aktivierter Checkbox erhalten nur die Kunden eine Bewertungserinnerung, " +"die ihr Einverständnis gegeben haben." + +#: includes/class-wc-trusted-shops-admin.php:649 +msgctxt "trusted-shops" +msgid "Edit checkbox" +msgstr "Checkbox anpassen" + +#: includes/class-wc-trusted-shops-admin.php:653 +msgctxt "trusted-shops" +msgid "Unsubscribe via link" +msgstr "Abmeldung per Link" + +#: includes/class-wc-trusted-shops-admin.php:654 +msgctxt "trusted-shops" +msgid "Allows the customer to unsubscribe from Review Requests." +msgstr "" +"Erlaubt es dem Kunden sich von Bewertungserinnerungen per Link abzumelden." + +#: includes/class-wc-trusted-shops-admin.php:755 +msgctxt "trusted-shops" +msgid "How does Trusted Shops make your shop better?" +msgstr "Wie macht Trusted Shops Ihren Shop besser?" + +#: includes/class-wc-trusted-shops-admin.php:757 +msgctxt "trusted-shops" +msgid "Get your account" +msgstr "Erstellen Sie Ihren Account" + +#: includes/class-wc-trusted-shops-admin.php:777 +msgctxt "trusted-shops" +msgid "Product Reviews on the product detail page in an additional tab" +msgstr "Produktbewertungen auf der Produktseite in einem separatem Reiter" + +#: includes/class-wc-trusted-shops-admin.php:780 +msgctxt "trusted-shops" +msgid "Show Star-Ratings on the product detail page below your product name" +msgstr "Bewertungssterne unter dem Produktnamen auf der Produktseite" + +#: includes/class-wc-trusted-shops-admin.php:784 +msgctxt "trusted-shops" +msgid "" +"Please note: If you want to send review requests through WooCommerce, you " +"should deactivate automated review requests through Trusted Shops. To do so, " +"please go to your My Trusted Shops account. Log in and go to Reviews > " +"Settings and deactivate \"Collect reviews automatically\"" +msgstr "" +"Bitte beachten Sie: Wenn Sie Bewertungserinnerungen über WooCommerce " +"versenden möchtest, sollten Sie den Versand von automatischen " +"Bewertungserinnerungen über Trusted Shops deaktivieren. Gehen Sie dazu in " +"Ihren My Trusted Shops Account. Loggen Sie sich ein und gehen Sie zu " +"Bewertungen > Konfiguration und deaktivieren Sie dort „Bewertungen " +"automatisch sammeln“." + +#: includes/class-wc-trusted-shops-admin.php:785 +msgctxt "trusted-shops" +msgid "To your My Trusted Shops account" +msgstr "Zu Ihrem My Trusted Shops Account" + +#: includes/class-wc-trusted-shops-admin.php:789 +msgctxt "trusted-shops" +msgid "" +"Export your customer information here and upload it in the Trusted Shops " +"Review Collector. To do so go to your My Trusted Shops account. Log in and " +"go to Reviews > Shop Reviews > Review Collector" +msgstr "" +"Exportieren Sie hier die Kundendaten und laden Sie diese im Trusted Shops " +"Review Collector hoch. Gehen Sie dazu in Ihren My Trusted Shops Account. " +"Loggen Sie sich ein und gehen Sie zu Bewertungen > Shopbewertungen > Review " +"Collector" + +#: includes/class-wc-trusted-shops-admin.php:790 +msgctxt "trusted-shops" +msgid "To the Trusted Shops Review Collector" +msgstr "Zum Trusted Shops Review Collector" + +#: includes/class-wc-trusted-shops-admin.php:880 +msgctxt "trusted-shops" +msgid "Review Collector" +msgstr "Review Collector" + +#: includes/class-wc-trusted-shops-admin.php:882 +#, php-format +msgctxt "trusted-shops" +msgid "" +"Want to collect reviews for orders that were placed before your Trusted " +"Shops Integration? No problem. Export old orders here and upload them in " +"your %s." +msgstr "" +"Sie möchten nachträglich Bewertungen zu Bestellungen sammeln, die vor der " +"Trusted Shops Integration getätigt wurden? Kein Problem. Exportieren Sie " +"alte Bestellungen hier und laden Sie diese in Ihrem %s hoch." + +#: includes/class-wc-trusted-shops-admin.php:882 +msgctxt "trusted-shops" +msgid "My Trusted Shops account" +msgstr "My Trusted Shops Account" + +#: includes/class-wc-trusted-shops-admin.php:888 +msgctxt "trusted-shops" +msgid "Export orders" +msgstr "Bestellungen exportieren" + +#: includes/class-wc-trusted-shops-admin.php:888 +msgctxt "trusted-shops" +msgid "" +"Export your customer and order information of the last x days and upload " +"them in your My Trusted Shops Account." +msgstr "" +"Exportieren Sie Ihre Kunden- und Bestelldaten der letzten x Tage und laden " +"Sie diese in Ihrem My Trusted Shops Account hoch." + +#: includes/class-wc-trusted-shops-admin.php:892 +msgctxt "trusted-shops" +msgid "30 days" +msgstr "30 Tage" + +#: includes/class-wc-trusted-shops-admin.php:893 +msgctxt "trusted-shops" +msgid "60 days" +msgstr "60 Tage" + +#: includes/class-wc-trusted-shops-admin.php:894 +msgctxt "trusted-shops" +msgid "90 days" +msgstr "90 Tage" + +#: includes/class-wc-trusted-shops-admin.php:896 +#, php-format +msgctxt "trusted-shops" +msgid "Upload customer and order information %s." +msgstr "Kunden- und Bestelldaten %s hochladen" + +#: includes/class-wc-trusted-shops-admin.php:899 +msgctxt "trusted-shops" +msgid "Days until reminder mail" +msgstr "Tage bis zur Erinnerung" + +#: includes/class-wc-trusted-shops-admin.php:899 +msgctxt "trusted-shops" +msgid "" +"Set the number of days to wait after the order date before having a Review " +"Request sent to your customers." +msgstr "" +"Stellen Sie hier ein, wie viele Tage zwischen der Bestellung und dem Versand " +"der Bewertungserinnerung liegen soll." + +#: includes/class-wc-trusted-shops-admin.php:903 +msgctxt "trusted-shops" +msgid "Start export" +msgstr "Export starten" + +#: includes/class-wc-trusted-shops-core.php:55 +#: includes/class-wc-trusted-shops-core.php:64 +#: includes/class-wc-ts-dependencies.php:36 +#: includes/class-wc-ts-dependencies.php:45 +msgctxt "trusted-shops" +msgid "Cheatin’ huh?" +msgstr "So geht das leider nicht.." + +#: includes/class-wc-trusted-shops-core.php:210 +msgctxt "trusted-shops" +msgid "Yes" +msgstr "Ja" + +#: includes/class-wc-trusted-shops-core.php:210 +msgctxt "trusted-shops" +msgid "No" +msgstr "Nein" + +#: includes/class-wc-trusted-shops-core.php:258 +#, php-format +msgctxt "trusted-shops" +msgid "" +"If the App helped you, please leave a %s★★" +"★★★%s in the Wordpress plugin repository." +msgstr "" +"Wenn Ihnen die App hilft, hinterlassen Sie bitte eine " +"%s★★★★★%s Bewertung in der WordPress Plugin " +"Bibliothek." + +#: includes/class-wc-trusted-shops-core.php:407 +msgctxt "trusted-shops" +msgid "Settings" +msgstr "Einstellungen" + +#: includes/class-wc-trusted-shops-review-exporter.php:64 +msgctxt "trusted-shops" +msgid "Order ID" +msgstr "Bestellnummer" + +#: includes/class-wc-trusted-shops-review-exporter.php:65 +msgctxt "trusted-shops" +msgid "Order date" +msgstr "Bestelldatum" + +#: includes/class-wc-trusted-shops-review-exporter.php:66 +msgctxt "trusted-shops" +msgid "# Days" +msgstr "# Tage" + +#: includes/class-wc-trusted-shops-review-exporter.php:67 +msgctxt "trusted-shops" +msgid "Email" +msgstr "E-Mail" + +#: includes/class-wc-trusted-shops-review-exporter.php:68 +msgctxt "trusted-shops" +msgid "First name" +msgstr "Vorname" + +#: includes/class-wc-trusted-shops-review-exporter.php:69 +msgctxt "trusted-shops" +msgid "Last name" +msgstr "Nachname" + +#: includes/class-wc-trusted-shops-template-hooks.php:121 +#, php-format +msgctxt "trusted-shops" +msgid "" +"Your review reminder e-mail has been cancelled successfully. Return to %s." +msgstr "" +"Ihre Bewertungserinnerung wurde erfolgreich deaktiviert. Kehren Sie zurück " +"zur %s." + +#: includes/class-wc-trusted-shops-template-hooks.php:121 +msgctxt "trusted-shops" +msgid "Home" +msgstr "Startseite" + +#: includes/class-wc-trusted-shops-template-hooks.php:193 +msgctxt "trusted-shops" +msgid "" +"Yes, I would like to be reminded via e-mail after {days} day(s) to review my " +"order. I am able to cancel the reminder at any time by clicking on the " +"\"cancel review reminder\" link within the order confirmation." +msgstr "" +"Ja, ich bin damit einverstanden, eine Erinnerung per E-Mail zur Bewertung " +"nach {days} Tage(n) zu erhalten. Ich kann mich jederzeit davon abmelden bzw. " +"widersprechen, indem ich dem Link „von der Bewertungserinnerung abmelden“ in " +"der Bestellbestätigung folge." + +#: includes/class-wc-trusted-shops-template-hooks.php:198 +msgctxt "trusted-shops" +msgid "Please allow us to send a review reminder by e-mail." +msgstr "" +"Bitte akzeptieren Sie den Erhalt einer Bewertungserinnerung per E-Mail." + +#: includes/class-wc-trusted-shops-template-hooks.php:201 +msgctxt "trusted-shops" +msgid "Review reminder" +msgstr "Bewertungs Erinnerung" + +#: includes/class-wc-trusted-shops-template-hooks.php:202 +msgctxt "trusted-shops" +msgid "Asks the customer to receive a Trusted Shops review reminder." +msgstr "" +"Holt die Erlaubnis zum Senden einer einmaligen Trusted Shops " +"Bewertungserinnerung ein." + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:24 +msgctxt "trusted-shops" +msgid "Trusted Shops Review Reminder" +msgstr "Trusted Shops Bewertungs-Erinnerung" + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:25 +msgctxt "trusted-shops" +msgid "" +"This E-Mail is being sent to a customer to remind him about the possibility " +"to leave a review at Trusted Shops." +msgstr "" +"Diese E-Mail wird einmalig an Kunden verschickt, um diese an die Abgabe " +"einer Bewertung bei Trusted Shops zu erinnern." + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:54 +msgctxt "trusted-shops" +msgid "Please rate your {site_title} order from {order_date}" +msgstr "Bitte bewerten Sie Ihren Einkauf vom {order_date} bei {site_title}" + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:64 +msgctxt "trusted-shops" +msgid "Please rate your Order" +msgstr "Bitte bewerten Sie Ihren Einkauf" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:19 +msgctxt "trusted-shops" +msgid "Show your TS shop review sticker." +msgstr "Zeige den Trusted Shops Review Sticker an." + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:21 +msgctxt "trusted-shops" +msgid "Trusted Shops Shop Review Sticker" +msgstr "Trusted Shops Shop Review Sticker" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:25 +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:46 +msgctxt "trusted-shops" +msgid "Trusted Shops Reviews" +msgstr "Trusted Shops Bewertung" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:26 +msgctxt "trusted-shops" +msgid "Title" +msgstr "Bezeichnung" + +#: src/Package.php:53 +msgctxt "trusted-shops" +msgid "" +"Trustbadge Reviews for WooCommerce needs at least WooCommerce version 3.1 to " +"run." +msgstr "" +"Trustbadge Reviews for WooCommerce benötigt mind. WooCommerce Version 3.1 um " +"zu starten." + +#: templates/emails/customer-trusted-shops.php:18 +#: templates/emails/plain/customer-trusted-shops.php:12 +#, php-format +msgctxt "trusted-shops" +msgid "Dear %s %s," +msgstr "Sehr geehrte(r) %s %s," + +#: templates/emails/customer-trusted-shops.php:19 +#: templates/emails/plain/customer-trusted-shops.php:14 +#, php-format +msgctxt "trusted-shops" +msgid "" +"You have recently shopped at %s. Thank you! We would be glad if you spent " +"some time to write a review about your order. To do so please follow follow " +"the link." +msgstr "" +"Sie haben vor einiger Zeit bei %s eingekauft. Vielen Dank! Wir wären froh " +"darüber, wenn Sie sich die Zeit nehmen und Ihren Einkauf bewerten würden. Um " +"dies nun zu tun, klicken Sie bitte auf den nachfolgenden Link." + +#: templates/emails/customer-trusted-shops.php:22 +msgctxt "trusted-shops" +msgid "Rate Order now" +msgstr "Einkauf jetzt bewerten" + +#~ msgctxt "trusted-shops" +#~ msgid "Duplicate Plugin installation" +#~ msgstr "Doppelte Plugin Installation" + +#, php-format +#~ msgctxt "trusted-shops" +#~ msgid "" +#~ "It seems like you've installed WooCommerce Germanized and Trustbadge " +#~ "Reviews for WooCommerce. Please deactivate Trustbadge Reviews for " +#~ "WooCommerce as long as you are using WooCommerce Germanized. You can " +#~ "manage your Trusted Shops configuration within your %s." +#~ msgstr "" +#~ "Es scheint als hättest du WooCommerce Germanized und Trustbadge Reviews " +#~ "for WooCommerce installiert. Bitte deaktiviere das Trustbadge Reviews " +#~ "Plugin solange du WooCommerce Germanized verwendest. Du kannst deine " +#~ "Trusted Shops Konfiguration in deinen %s verwalten." + +#~ msgctxt "trusted-shops" +#~ msgid "Germanized settings" +#~ msgstr "Germanized Einstellungen" + +#~ msgctxt "trusted-shops" +#~ msgid "Deactivate standalone version" +#~ msgstr "Deaktiviere die Standalone-Version" + +#~ msgctxt "trusted-shops" +#~ msgid "(WooCommerce Product Reviews will be replaced)" +#~ msgstr "(WooCommerce Produktbewertungen werden ersetzt)" + +#, php-format +#~ msgid "" +#~ "Please install WooCommerce before " +#~ "installing WooCommerce Germanized. Thank you!" +#~ msgstr "" +#~ "Bitte installiere WooCommerce bevor " +#~ "du WooCommerce Germanized installierst. Vielen Dank!" diff --git a/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-fr_FR.mo b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-fr_FR.mo new file mode 100644 index 000000000..19ef22d77 Binary files /dev/null and b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-fr_FR.mo differ diff --git a/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-fr_FR.po b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-fr_FR.po new file mode 100644 index 000000000..7e00b75f8 --- /dev/null +++ b/packages/woocommerce-trusted-shops/i18n/languages/woocommerce-trusted-shops-fr_FR.po @@ -0,0 +1,1165 @@ +msgid "" +msgstr "" +"Project-Id-Version: WooCommerce Trusted Shops\n" +"POT-Creation-Date: 2019-01-14 15:41+0100\n" +"PO-Revision-Date: 2019-02-18 15:10+0100\n" +"Last-Translator: \n" +"Language-Team: \n" +"Language: fr\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"X-Generator: Poedit 2.2.1\n" +"X-Poedit-Basepath: ../..\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Poedit-KeywordsList: __;_e;__ngettext:1,2;_n:1,2;__ngettext_noop:1,2;" +"_n_noop:1,2;_c,_nc:4c,1,2;_x:1,2c;_ex:1,2c;_nx:4c,1,2;_nx_noop:4c,1,2;" +"esc_attr_e;esc_html_e;esc_html__\n" +"X-Poedit-SearchPath-0: .\n" +"X-Poedit-SearchPathExcluded-0: node_modules\n" + +#: includes/admin/views/html-duplicate-plugin-notice.php:19 +msgctxt "trusted-shops" +msgid "Duplicate Plugin installation" +msgstr "Plugins installés en double" + +#: includes/admin/views/html-duplicate-plugin-notice.php:21 +#, php-format +msgctxt "trusted-shops" +msgid "" +"It seems like you've installed WooCommerce Germanized and Trustbadge Reviews " +"for WooCommerce. Please deactivate Trustbadge Reviews for WooCommerce as " +"long as you are using WooCommerce Germanized. You can manage your Trusted " +"Shops configuration within your %s." +msgstr "" +"Il semblerait que vous ayez installé WooCommerce Germanized et Trustbadge " +"Reviews for WooCommerce. Veuillez s’il vous plait désactiver Trustbadge " +"Reviews for WooCommerce tant que vous utilisez WooCommerce Germanized. Vous " +"pouvez modifier votre configuration Trusted Shops au sein de vos %s." + +#: includes/admin/views/html-duplicate-plugin-notice.php:21 +msgctxt "trusted-shops" +msgid "Germanized settings" +msgstr "Germanized paramètres" + +#: includes/admin/views/html-duplicate-plugin-notice.php:23 +msgctxt "trusted-shops" +msgid "Deactivate standalone version" +msgstr "Désactivez la version standalone" + +#: includes/admin/views/html-notice-dependencies.php:16 +msgctxt "trusted-shops" +msgid "Dependencies Missing or Outdated" +msgstr "Composants manquantes ou obsolètes" + +#: includes/admin/views/html-notice-dependencies.php:24 +msgctxt "trusted-shops" +msgid "" +"To use WooCommerce Trusted Shops you may at first install the following " +"plugins:" +msgstr "" +"Pour utiliser Trusted Shops for WooCommerce, il vous faut d’abord installer " +"les plugins suivants :" + +#: includes/admin/views/html-notice-dependencies.php:28 +#, php-format +msgctxt "trusted-shops" +msgid "Install %s" +msgstr "Installer %s" + +#: includes/admin/views/html-notice-dependencies.php:36 +msgctxt "trusted-shops" +msgid "" +"To use WooCommerce Trusted Shops you may at first update the following " +"plugins to a newer version:" +msgstr "" +"Pour utiliser Trusted Shops for WooCommerce, vous devez d’abord actualiser " +"les plugins suivants et installer une version plus récente:" + +#: includes/admin/views/html-notice-dependencies.php:40 +#, php-format +msgctxt "trusted-shops" +msgid "%s required in at least version %s" +msgstr "%s requiert la version %s ou supérieure" + +#: includes/admin/views/html-notice-dependencies.php:49 +msgctxt "trusted-shops" +msgid "Check for Updates" +msgstr "Rechercher de mises à jour" + +#: includes/admin/views/html-notice-dependencies.php:50 +msgctxt "trusted-shops" +msgid "or" +msgstr "ou" + +#: includes/admin/views/html-notice-dependencies.php:51 +msgctxt "trusted-shops" +msgid "Install an older version" +msgstr "Installer une version antérieure" + +#: includes/admin/views/html-notice-update.php:12 +msgctxt "trusted-shops" +msgid "" +"WooCommerce Trusted Shops Data Update Required – We " +"just need to update your install to the latest version" +msgstr "" +"Mise à jour de la base de données Trusted Shops WooCommerce requise – Nous devons mettre à jour votre installation vers la version " +"la plus récente." + +#: includes/admin/views/html-notice-update.php:13 +msgctxt "trusted-shops" +msgid "Run the updater" +msgstr "Démarrer l’actualisation" + +#: includes/admin/views/html-notice-update.php:17 +msgctxt "trusted-shops" +msgid "" +"It is strongly recommended that you backup your database before proceeding. " +"Are you sure you wish to run the updater now?" +msgstr "" +"Nous vous recommandons fortement de sauvegarder votre base de données avant " +"de procéder à la mise à jour. Êtes-vous sûr de vouloir effectuer la mise à " +"jour maintenant?" + +#: includes/admin/views/html-wpml-notice.php:11 +msgctxt "trusted-shops" +msgid "WPML Support" +msgstr "Prise en charge WPML" + +#: includes/admin/views/html-wpml-notice.php:14 +msgctxt "trusted-shops" +msgid "" +"These settings serve as default settings for all your languages. To adjust " +"the settings for a certain language, please switch your admin language " +"through the WPML language switcher and adjust the corresponding settings." +msgstr "" +"Ces paramètres sont les paramètres par défaut pour toutes vos langues. Afin " +"d’ajuster ces paramètres à une certaine langue, il vous faut changer votre " +"langue système à l’aide du plug-in multilingue WPML et adapter les " +"paramètres correspondants." + +#: includes/admin/views/html-wpml-notice.php:16 +#, php-format +msgctxt "trusted-shops" +msgid "" +"These settings apply for your %s shop. To adjust settings for another " +"language, please switch your admin language through the WPML language " +"switcher." +msgstr "" +"Ces paramètres s’appliquent pour votre boutique %s. Pour adapter les " +"paramètres à une autre langue, veuillez changer votre langue système à " +"l’aide du plug-in multilingue WPML." + +#: includes/class-wc-trusted-shops-admin.php:90 +#: includes/class-wc-trusted-shops-admin.php:127 +msgctxt "trusted-shops" +msgid "GTIN" +msgstr "GTIN" + +#: includes/class-wc-trusted-shops-admin.php:90 +#: includes/class-wc-trusted-shops-admin.php:127 +msgctxt "trusted-shops" +msgid "" +"ID that allows your products to be identified worldwide. If you want to " +"display your Trusted Shops Product Reviews in Google Shopping and paid " +"Google adverts, Google needs the GTIN." +msgstr "" +"Le GTIN est le numéro d’identification qui permet d’identifier sans " +"équivoque les produits dans le monde entier. Google a besoin de l’attribut " +"GTIN pour afficher vos avis produits dans Google Shopping et dans les " +"annonces payantes Google." + +#: includes/class-wc-trusted-shops-admin.php:94 +#: includes/class-wc-trusted-shops-admin.php:128 +msgctxt "trusted-shops" +msgid "MPN" +msgstr "MPN" + +#: includes/class-wc-trusted-shops-admin.php:94 +#: includes/class-wc-trusted-shops-admin.php:128 +msgctxt "trusted-shops" +msgid "" +"If you don't have a GTIN for your products, you can pass the brand name and " +"the MPN on to Google to use the Trusted Shops Google Integration." +msgstr "" +"Si vos produits n’ont pas de GTIN, vous pouvez indiquer le nom de la marque " +"ainsi que le MPN pour utiliser l’intégration Google." + +#: includes/class-wc-trusted-shops-admin.php:152 +msgctxt "trusted-shops" +msgid "Brand" +msgstr "Marque" + +#: includes/class-wc-trusted-shops-admin.php:192 +msgctxt "trusted-shops" +msgid "This field is mandatory" +msgstr "Ce champ est obligatoire" + +#: includes/class-wc-trusted-shops-admin.php:199 +msgctxt "trusted-shops" +msgid "Trusted Shops Options" +msgstr "Options Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:206 +msgctxt "trusted-shops" +msgid "Arial" +msgstr "Arial" + +#: includes/class-wc-trusted-shops-admin.php:207 +msgctxt "trusted-shops" +msgid "Geneva" +msgstr "Geneva" + +#: includes/class-wc-trusted-shops-admin.php:208 +msgctxt "trusted-shops" +msgid "Georgia" +msgstr "Georgia" + +#: includes/class-wc-trusted-shops-admin.php:209 +msgctxt "trusted-shops" +msgid "Helvetica" +msgstr "Helvetica" + +#: includes/class-wc-trusted-shops-admin.php:210 +msgctxt "trusted-shops" +msgid "Sans-serif" +msgstr "Sans-serif" + +#: includes/class-wc-trusted-shops-admin.php:211 +msgctxt "trusted-shops" +msgid "Serif" +msgstr "Serif" + +#: includes/class-wc-trusted-shops-admin.php:212 +msgctxt "trusted-shops" +msgid "Trebuchet MS" +msgstr "Trebuchet MS" + +#: includes/class-wc-trusted-shops-admin.php:213 +msgctxt "trusted-shops" +msgid "Verdana" +msgstr "Verdana" + +#: includes/class-wc-trusted-shops-admin.php:236 +msgctxt "trusted-shops" +msgid "Trusted Shops Integration" +msgstr "Intégration Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:237 +#, php-format +msgctxt "trusted-shops" +msgid "Do you need help with integrating your Trustbadge? %s" +msgstr "Avez-vous besoin d’aide pour intégrer le Trustbadge? %s" + +#: includes/class-wc-trusted-shops-admin.php:237 +msgctxt "trusted-shops" +msgid "To the step-by-step instructions" +msgstr "Accéder au guide" + +#: includes/class-wc-trusted-shops-admin.php:243 +msgctxt "trusted-shops" +msgid "Trusted Shops ID" +msgstr "Identifiants Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:244 +msgctxt "trusted-shops" +msgid "" +"The Trusted Shops ID is a unique identifier for your shop. You can find your " +"Trusted Shops ID in your My Trusted Shops account." +msgstr "" +"Les identifiants Trusted Shops servent à identifier votre boutique en ligne " +"sur Trusted Shops. Vous trouverez vos identifiants Trusted Shops (TSID) dans " +"votre compte My Trusted Shops." + +#: includes/class-wc-trusted-shops-admin.php:253 +msgctxt "trusted-shops" +msgid "Edit Mode" +msgstr "Mode édition" + +#: includes/class-wc-trusted-shops-admin.php:255 +#: includes/class-wc-trusted-shops-admin.php:316 +#: includes/class-wc-trusted-shops-admin.php:439 +msgctxt "trusted-shops" +msgid "" +"The advanced configuration is for users with programming skills. Here you " +"can create even more individual settings." +msgstr "" +"Le mode expert est conçu pour des utilisateurs ayant de bonnes connaissances " +"en programmation. Cliquer ici, pour avoir accès à des réglages personnalisés." + +#: includes/class-wc-trusted-shops-admin.php:259 +msgctxt "trusted-shops" +msgid "Standard configuration" +msgstr "Mode standard" + +#: includes/class-wc-trusted-shops-admin.php:260 +msgctxt "trusted-shops" +msgid "Advanced configuration" +msgstr "Mode expert" + +#: includes/class-wc-trusted-shops-admin.php:268 +msgctxt "trusted-shops" +msgid "Configure your Trustbadge" +msgstr "Configurer le Trustbadge" + +#: includes/class-wc-trusted-shops-admin.php:274 +msgctxt "trusted-shops" +msgid "Display Trustbadge" +msgstr "Afficher le Trustbadge" + +#: includes/class-wc-trusted-shops-admin.php:276 +msgctxt "trusted-shops" +msgid "Display the Trustbadge on all the pages of your shop." +msgstr "" +"Afficher le Trustbadge sur toutes les pages de votre boutique en ligne." + +#: includes/class-wc-trusted-shops-admin.php:283 +msgctxt "trusted-shops" +msgid "Variant" +msgstr "Variante" + +#: includes/class-wc-trusted-shops-admin.php:285 +msgctxt "trusted-shops" +msgid "You can display your Trustbadge with or without Review Stars." +msgstr "" +"Vous pouvez afficher le Trustbadge avec ou sans les étoiles d’évaluation." + +#: includes/class-wc-trusted-shops-admin.php:289 +#: includes/class-wc-trusted-shops-admin.php:780 +msgctxt "trusted-shops" +msgid "Display Trustbadge with review stars" +msgstr "Afficher le Trustbadge avec les étoiles d’évaluation" + +#: includes/class-wc-trusted-shops-admin.php:290 +#: includes/class-wc-trusted-shops-admin.php:784 +msgctxt "trusted-shops" +msgid "Display Trustbadge without review stars" +msgstr "Afficher le Trustbadge sans les étoiles d’évaluation" + +#: includes/class-wc-trusted-shops-admin.php:296 +msgctxt "trusted-shops" +msgid "Vertical Offset" +msgstr "Décalage vertical" + +#: includes/class-wc-trusted-shops-admin.php:297 +msgctxt "trusted-shops" +msgid "" +"Choose the distance that the Trustbadge will appear from the bottom-right " +"corner of the screen." +msgstr "" +"Choisissez l’espacement du Trustbadge® par rapport au bord inferieur droit " +"de l’écran." + +#: includes/class-wc-trusted-shops-admin.php:300 +#: includes/class-wc-trusted-shops-admin.php:509 +#: includes/class-wc-trusted-shops-admin.php:560 +#: includes/class-wc-trusted-shops-admin.php:575 +msgid "px" +msgstr "px" + +#: includes/class-wc-trusted-shops-admin.php:306 +#: includes/class-wc-trusted-shops-admin.php:516 +#: includes/class-wc-trusted-shops-admin.php:566 +#: includes/class-wc-trusted-shops-admin.php:582 +#, php-format +msgctxt "trusted-shops" +msgid "Please choose a non-negative number (at least %d)" +msgstr "Veuillez choisir un nombre positif (> %d)" + +#: includes/class-wc-trusted-shops-admin.php:312 +msgctxt "trusted-shops" +msgid "Trustbadge code" +msgstr "Trustbadge code" + +#: includes/class-wc-trusted-shops-admin.php:324 +msgctxt "trusted-shops" +msgid "Configure your Shop Reviews" +msgstr "Configurer les avis site" + +#: includes/class-wc-trusted-shops-admin.php:330 +msgctxt "trusted-shops" +msgid "Display Shop Review Sticker" +msgstr "Afficher la vignette avis site" + +#: includes/class-wc-trusted-shops-admin.php:331 +msgctxt "trusted-shops" +msgid "" +"To display the Shop Review Sticker, you have to assign the widget \"Trusted " +"Shops Review Sticker\"." +msgstr "" +"Pour afficher la vignette avis site, veuillez sélectionner le widget " +"« Vignette avis site Trusted Shops »." + +#: includes/class-wc-trusted-shops-admin.php:332 +#, php-format +msgctxt "trusted-shops" +msgid "Assign widget %s" +msgstr "Assigner le widget %s" + +#: includes/class-wc-trusted-shops-admin.php:332 +#: includes/class-wc-trusted-shops-admin.php:605 +#: includes/class-wc-trusted-shops-admin.php:905 +msgctxt "trusted-shops" +msgid "here" +msgstr "ici" + +#: includes/class-wc-trusted-shops-admin.php:340 +#: includes/class-wc-trusted-shops-admin.php:489 +msgctxt "trusted-shops" +msgid "Background color" +msgstr "Couleur de l’arrière-plan" + +#: includes/class-wc-trusted-shops-admin.php:341 +msgctxt "trusted-shops" +msgid "Choose the background color for your Review Sticker." +msgstr "Choisir la couleur de l’arrière-plan de la vignette d’avis." + +#: includes/class-wc-trusted-shops-admin.php:348 +msgctxt "trusted-shops" +msgid "Font" +msgstr "Police" + +#: includes/class-wc-trusted-shops-admin.php:350 +msgctxt "trusted-shops" +msgid "Choose the font for your Review Sticker." +msgstr "Choisir la police de la vignette d’avis." + +#: includes/class-wc-trusted-shops-admin.php:357 +msgctxt "trusted-shops" +msgid "Number of reviews displayed" +msgstr "Nombre d’avis" + +#: includes/class-wc-trusted-shops-admin.php:358 +msgctxt "trusted-shops" +msgid "" +"Display x alternating Shop Reviews in your Shop Review Sticker. You can " +"display between 1 and 5 alternating Shop Reviews." +msgstr "" +"Afficher x avis site en alternance sur votre vignette avis site. Vous pouvez " +"afficher en alternance de 1 à 5 avis." + +#: includes/class-wc-trusted-shops-admin.php:361 +msgctxt "trusted-shops" +msgid "Show x alternating reviews" +msgstr "Afficher x avis en alternance" + +#: includes/class-wc-trusted-shops-admin.php:368 +#: includes/class-wc-trusted-shops-admin.php:385 +#, php-format +msgctxt "trusted-shops" +msgid "Please choose a non-negative number between %d and %d" +msgstr "Veuillez choisir un nombre positif entre %d et %d" + +#: includes/class-wc-trusted-shops-admin.php:374 +msgctxt "trusted-shops" +msgid "Minimum rating displayed" +msgstr "Note minimale affichée" + +#: includes/class-wc-trusted-shops-admin.php:375 +msgctxt "trusted-shops" +msgid "Only show Shop Reviews with a minimum rating of x stars. " +msgstr "N’afficher que les avis site avec x étoiles ou plus. " + +#: includes/class-wc-trusted-shops-admin.php:378 +msgctxt "trusted-shops" +msgid "Star(s)" +msgstr "étoile(s)" + +#: includes/class-wc-trusted-shops-admin.php:391 +msgctxt "trusted-shops" +msgid "Sticker code" +msgstr "Code des vignettes avis clients" + +#: includes/class-wc-trusted-shops-admin.php:395 +#: includes/class-wc-trusted-shops-admin.php:523 +#: includes/class-wc-trusted-shops-admin.php:588 +msgctxt "trusted-shops" +msgid "" +"The advanced configuration is for users with programming skills. Here you " +"can perform even more individual settings." +msgstr "" +"Le mode expert est conçu pour des utilisateurs ayant de bonnes connaissances " +"en programmation. Cliquer ici, pour avoir accès à des réglages personnalisés." + +#: includes/class-wc-trusted-shops-admin.php:401 +msgctxt "trusted-shops" +msgid "Google Organic Search" +msgstr "Résultats de recherche organiques" + +#: includes/class-wc-trusted-shops-admin.php:402 +msgctxt "trusted-shops" +msgid "" +"Activate this option to give Google the opportunity to show your Shop " +"Reviews in Google organic search results." +msgstr "" +"Veuillez activer cette fonctionnalité pour permettre à Google d’afficher vos " +"avis site dans les résultats de recherche organiques." + +#: includes/class-wc-trusted-shops-admin.php:403 +msgctxt "trusted-shops" +msgid "" +"By activating this option, rich snippets will be integrated in the selected " +"pages so your shop review stars may be displayed in Google organic search " +"results. If you use Product Reviews and already activated rich snippets in " +"expert mode, we recommend integrating rich snippets for Shop Reviews on " +"category pages only." +msgstr "" +"En activant cette option, les rich snippets seront intégrés dans les pages " +"sélectionnées, de façon à ce que vos étoiles d’évaluation puissent être " +"affichées dans les résultats de recherche organiques de Google. Si vous avez " +"activé les avis produits et vous avez déjà activé les rich snippets dans le " +"mode expert, nous vous conseillons de n’activer les rich snippets pour les " +"avis site que sur la page de catégorie." + +#: includes/class-wc-trusted-shops-admin.php:410 +msgctxt "trusted-shops" +msgid "Activate rich snippets on" +msgstr "Activer les rich snippets sur" + +#: includes/class-wc-trusted-shops-admin.php:411 +msgctxt "trusted-shops" +msgid "category pages" +msgstr "les pages de catégories" + +#: includes/class-wc-trusted-shops-admin.php:419 +msgctxt "trusted-shops" +msgid "product pages" +msgstr "les pages produit" + +#: includes/class-wc-trusted-shops-admin.php:427 +msgctxt "trusted-shops" +msgid "homepage (not recommended)" +msgstr "la page d’accueil (non conseillé)" + +#: includes/class-wc-trusted-shops-admin.php:435 +msgctxt "trusted-shops" +msgid "Rich snippets code" +msgstr "Code pour les Rich Snippets" + +#: includes/class-wc-trusted-shops-admin.php:447 +msgctxt "trusted-shops" +msgid "Configure your Product Reviews " +msgstr "Configurer les avis produits " + +#: includes/class-wc-trusted-shops-admin.php:448 +#, php-format +msgctxt "trusted-shops" +msgid "To use Product Reviews, activate them in your %s first." +msgstr "" +"Pour pouvoir utiliser les avis produits, veuillez d’abord activer l’option " +"%s." + +#: includes/class-wc-trusted-shops-admin.php:448 +msgctxt "trusted-shops" +msgid "Trusted Shops package" +msgstr "Services Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:454 +msgctxt "trusted-shops" +msgid "Collect Product Reviews" +msgstr "Collecter des avis produits" + +#: includes/class-wc-trusted-shops-admin.php:455 +msgctxt "trusted-shops" +msgid "(WooCommerce Product Reviews will be replaced)" +msgstr "(Les avis produits WooCommerce seront remplacés)" + +#: includes/class-wc-trusted-shops-admin.php:456 +msgctxt "trusted-shops" +msgid "" +"Show Product Reviews on the product page in a separate tab, just as shown on " +"the picture on the right." +msgstr "Afficher les avis produits dans un onglet séparé sur la page produit." + +#: includes/class-wc-trusted-shops-admin.php:464 +msgctxt "trusted-shops" +msgid "Reviews" +msgstr "Avis clients" + +#: includes/class-wc-trusted-shops-admin.php:465 +#: includes/class-wc-trusted-shops-admin.php:474 +msgctxt "trusted-shops" +msgid "You can choose a name for the tab with your Product Reviews." +msgstr "" +"Vous pouvez choisir le nom de l’onglet dans lequel vous souhaitez afficher " +"vos avis produits." + +#: includes/class-wc-trusted-shops-admin.php:466 +msgctxt "trusted-shops" +msgid "Show Product Reviews on the product detail page in an additional tab." +msgstr "Afficher les avis produits dans un onglet séparé sur la page produit." + +#: includes/class-wc-trusted-shops-admin.php:473 +msgctxt "trusted-shops" +msgid "Name of Product Reviews tab" +msgstr "Nom de l’onglet" + +#: includes/class-wc-trusted-shops-admin.php:477 +msgctxt "trusted-shops" +msgid "Product reviews" +msgstr "Avis produits" + +#: includes/class-wc-trusted-shops-admin.php:481 +msgctxt "trusted-shops" +msgid "Border color" +msgstr "Couleur du cadre" + +#: includes/class-wc-trusted-shops-admin.php:482 +msgctxt "trusted-shops" +msgid "Set the color for the frame around your Product Reviews." +msgstr "Choisir la couleur du cadre des avis produits." + +#: includes/class-wc-trusted-shops-admin.php:490 +msgctxt "trusted-shops" +msgid "Set the background color for your Product Reviews." +msgstr "Choisir la couleur de l’arrière-plan des avis produits." + +#: includes/class-wc-trusted-shops-admin.php:497 +#: includes/class-wc-trusted-shops-admin.php:547 +msgctxt "trusted-shops" +msgid "Star color" +msgstr "Couleur des étoiles" + +#: includes/class-wc-trusted-shops-admin.php:498 +msgctxt "trusted-shops" +msgid "Set the color for the Product Review stars in your Product Reviews tab." +msgstr "" +"Choisi la couleur des étoiles qui seront affichées dans l’onglet avec les " +"avis produits." + +#: includes/class-wc-trusted-shops-admin.php:505 +#: includes/class-wc-trusted-shops-admin.php:555 +msgctxt "trusted-shops" +msgid "Star size" +msgstr "Dimension des étoiles" + +#: includes/class-wc-trusted-shops-admin.php:510 +msgctxt "trusted-shops" +msgid "Set the size for the Product Review stars in your Product Reviews tab." +msgstr "" +"Choisir la dimension des étoiles qui seront affichées dans l’onglet avec les " +"avis produits." + +#: includes/class-wc-trusted-shops-admin.php:521 +msgctxt "trusted-shops" +msgid "Product Sticker Code" +msgstr "Code des vignettes avis produits" + +#: includes/class-wc-trusted-shops-admin.php:530 +#: includes/class-wc-trusted-shops-admin.php:596 +msgctxt "trusted-shops" +msgid "jQuerySelector" +msgstr "jQuerySelector" + +#: includes/class-wc-trusted-shops-admin.php:531 +msgctxt "trusted-shops" +msgid "" +"Please choose where your Product Reviews shall be displayed on the Product " +"detail page." +msgstr "" +"Veuillez choisir l’endroit où les Avis Produits seront affichés sur la page " +"produit" + +#: includes/class-wc-trusted-shops-admin.php:538 +msgctxt "trusted-shops" +msgid "Rating stars" +msgstr "Étoiles d’évaluation" + +#: includes/class-wc-trusted-shops-admin.php:539 +msgctxt "trusted-shops" +msgid "Show star ratings on the product detail page below your product name." +msgstr "" +"Afficher les étoiles d’évaluation sur la page produit en dessous du nom du " +"produit." + +#: includes/class-wc-trusted-shops-admin.php:540 +msgctxt "trusted-shops" +msgid "" +"Display Product Review stars on product pages below the product name, just " +"as shown in the picture on the right." +msgstr "" +"Afficher les étoiles d’évaluation sur la page produit en dessous du nom du " +"produit, comme illustré par l’image ci-contre." + +#: includes/class-wc-trusted-shops-admin.php:549 +msgctxt "trusted-shops" +msgid "" +"Set the color for the review stars, that are displayed on the product page, " +"below your product name." +msgstr "" +"Choisir la couleur des étoiles qui seront affichées sur la page produit en " +"dessous du nom du produit." + +#: includes/class-wc-trusted-shops-admin.php:557 +msgctxt "trusted-shops" +msgid "" +"Set the size for the review stars that are displayed on the product page, " +"below your product name." +msgstr "" +"Choisir la dimension des étoiles qui seront affichées sur la page produit en " +"dessous du nom du produit." + +#: includes/class-wc-trusted-shops-admin.php:571 +msgctxt "trusted-shops" +msgid "Font size" +msgstr "Taille de la police" + +#: includes/class-wc-trusted-shops-admin.php:573 +msgctxt "trusted-shops" +msgid "Set the font size for the text that goes with your review stars." +msgstr "" +"Choisir la taille de la police du texte correspondant à vos étoiles " +"d’évaluation." + +#: includes/class-wc-trusted-shops-admin.php:587 +msgctxt "trusted-shops" +msgid "Product Review Code" +msgstr "Code pour les étoiles d’évaluation" + +#: includes/class-wc-trusted-shops-admin.php:597 +msgctxt "trusted-shops" +msgid "" +"Please choose where your Product Review Stars shall be displayed on the " +"Product Detail page." +msgstr "" +"Veuillez choisir l’endroit où les étoiles des Avis Produit seront affichées " +"sur la page produit" + +#: includes/class-wc-trusted-shops-admin.php:604 +msgctxt "trusted-shops" +msgid "Brand attribute" +msgstr "Attribut marque" + +#: includes/class-wc-trusted-shops-admin.php:605 +#, php-format +msgctxt "trusted-shops" +msgid "Create brand attribute %s" +msgstr "Créer %s un attribut marque" + +#: includes/class-wc-trusted-shops-admin.php:606 +msgctxt "trusted-shops" +msgid "" +"Brand name of the product. By passing this information on to Google, you " +"improve your chances of having Google identify your products. Assign your " +"brand attribute. If your products don't have a GTIN, you can pass on the " +"brand name and the MPN to use Google Integration." +msgstr "" +"Nom de la marque du produit. En indiquant cette variable, il est plus " +"probable que Google identifie clairement vos produits. Veuillez assigner " +"l’attribut marque ici. Si vos produits n’ont pas de GTIN, vous pouvez " +"indiquer le nom de la marque ainsi que le MPN pour utiliser l’intégration " +"Google." + +#: includes/class-wc-trusted-shops-admin.php:612 +msgctxt "trusted-shops" +msgid "None" +msgstr "Aucun" + +#: includes/class-wc-trusted-shops-admin.php:623 +msgctxt "trusted-shops" +msgid "Configure your Review Requests" +msgstr "Configurer vos rappels d’avis" + +#: includes/class-wc-trusted-shops-admin.php:624 +msgctxt "trusted-shops" +msgid "" +"7 days after an order has been placed, Trusted Shops automatically sends an " +"invite to your customers. If you want to set a different time for sending " +"automatic Review Requests, please activate the option below. If you want to " +"send review requests with legal certainty, you need your customers' consent " +"to receive Review Requests. You also have to include an option to " +"unsubscribe." +msgstr "" +"Trusted Shops envoie automatiquement un rappel d’avis à vos clients 7 jours " +"après leur commande. Si vous préférez décider vous-même du moment auquel " +"vous souhaitez envoyer vos demandes d’avis, cochez l’option ci-dessous. Pour " +"envoyer des rappels d’avis en conformité avec la loi, il faut obtenir le " +"consentement exprès des clients à la réception d’e-mails de demande d’avis. " +"Vous devez également donner à vos clients un moyen de se désabonner des " +"rappels d’avis." + +#: includes/class-wc-trusted-shops-admin.php:630 +msgctxt "trusted-shops" +msgid "Enable Review Requests" +msgstr "Activer les demandes d’avis" + +#: includes/class-wc-trusted-shops-admin.php:639 +msgctxt "trusted-shops" +msgid "WooCommerce status" +msgstr "État de la commande" + +#: includes/class-wc-trusted-shops-admin.php:640 +msgctxt "trusted-shops" +msgid "" +"We recommend choosing the order status that you set when your products have " +"been shipped." +msgstr "" +"Sélectionner l’état de la commande que vous avez défini au moment de l’envoi " +"de la commande." + +#: includes/class-wc-trusted-shops-admin.php:648 +msgctxt "trusted-shops" +msgid "Days until Review Request" +msgstr "Jours avant envoi du rappel" + +#: includes/class-wc-trusted-shops-admin.php:649 +msgctxt "trusted-shops" +msgid "" +"Set the number of days to wait after an order has reached the order status " +"you selected above before having a review request sent to your customers." +msgstr "" +"Indiquer ici après combien de jours le rappel d’évaluation est envoyé à " +"l’acheteur après avoir atteint l’état de la commande sélectionné ci-dessus." + +#: includes/class-wc-trusted-shops-admin.php:661 +msgctxt "trusted-shops" +msgid "Permission via checkbox" +msgstr "Consentement par case à cocher" + +#: includes/class-wc-trusted-shops-admin.php:662 +msgctxt "trusted-shops" +msgid "" +"If the checkbox is activated, only customers who gave their consent will " +"receive Review Requests." +msgstr "" +"Si la case est cochée, seuls les clients ayant donné leur consentement " +"recevront un rappel d’avis par e-mail." + +#: includes/class-wc-trusted-shops-admin.php:666 +msgctxt "trusted-shops" +msgid "Edit checkbox" +msgstr "Modifier la case à cocher" + +#: includes/class-wc-trusted-shops-admin.php:670 +msgctxt "trusted-shops" +msgid "Unsubscribe via link" +msgstr "Désinscription via lien" + +#: includes/class-wc-trusted-shops-admin.php:671 +msgctxt "trusted-shops" +msgid "Allows the customer to unsubscribe from Review Requests." +msgstr "" +"Cette option permet aux clients de se désabonner des rappels d’avis en " +"cliquant sur un lien." + +#: includes/class-wc-trusted-shops-admin.php:772 +msgctxt "trusted-shops" +msgid "How does Trusted Shops make your shop better?" +msgstr "" +"En quoi les produits Trusted Shops contribuent-ils à optimiser votre " +"boutique en ligne?" + +#: includes/class-wc-trusted-shops-admin.php:774 +msgctxt "trusted-shops" +msgid "Get your account" +msgstr "Créer un compte" + +#: includes/class-wc-trusted-shops-admin.php:794 +msgctxt "trusted-shops" +msgid "Product Reviews on the product detail page in an additional tab" +msgstr "Avis produits dans l’onglet sur la page produit" + +#: includes/class-wc-trusted-shops-admin.php:797 +msgctxt "trusted-shops" +msgid "Show Star-Ratings on the product detail page below your product name" +msgstr "Étoiles d’évaluation sur la page produit en dessous du nom du produit" + +#: includes/class-wc-trusted-shops-admin.php:801 +msgctxt "trusted-shops" +msgid "" +"Please note: If you want to send review requests through WooCommerce, you " +"should deactivate automated review requests through Trusted Shops. To do so, " +"please go to your My Trusted Shops account. Log in and go to Reviews > " +"Settings and deactivate \"Collect reviews automatically\"" +msgstr "" +"Veuillez noter : Si vous souhaitez envoyer des rappels d’avis via " +"WooCommerce, vous devez désactiver l’envoi de rappels d’avis automatiques " +"via Trusted Shops. Pour ce faire, rendez-vous dans votre compte My Trusted " +"Shops. Connectez-vous à votre compte, cliquez sur Avis clients > " +"Configuration > Collecter les avis et désactivez l’option « Collecter des " +"avis automatiquement »." + +#: includes/class-wc-trusted-shops-admin.php:802 +msgctxt "trusted-shops" +msgid "To your My Trusted Shops account" +msgstr "Aller au compte My Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:806 +msgctxt "trusted-shops" +msgid "" +"Export your customer information here and upload it in the Trusted Shops " +"Review Collector. To do so go to your My Trusted Shops account. Log in and " +"go to Reviews > Shop Reviews > Review Collector" +msgstr "" +"Exporter les données de vos clients et de vos commandes des x derniers jours " +"et importez-les dans votre compte My Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:807 +msgctxt "trusted-shops" +msgid "To the Trusted Shops Review Collector" +msgstr "Aller au collecteur d’avis Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:889 +msgctxt "trusted-shops" +msgid "Review Collector" +msgstr "Collecteur d’avis Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:891 +#, php-format +msgctxt "trusted-shops" +msgid "" +"Want to collect reviews for orders that were placed before your Trusted " +"Shops Integration? No problem. Export old orders here and upload them in " +"your %s." +msgstr "" +"Vous souhaitez demander à des clients de laisser un avis sur des commandes " +"qu’ils ont passées avant l’intégration de Trusted Shops sur votre site ? Pas " +"de problème! Exportez les commandes passées et importez-les dans votre %s." + +#: includes/class-wc-trusted-shops-admin.php:891 +msgctxt "trusted-shops" +msgid "My Trusted Shops account" +msgstr "Compte My Trusted Shops" + +#: includes/class-wc-trusted-shops-admin.php:897 +msgctxt "trusted-shops" +msgid "Export orders" +msgstr "Exporter les commandes" + +#: includes/class-wc-trusted-shops-admin.php:897 +msgctxt "trusted-shops" +msgid "" +"Export your customer and order information of the last x days and upload " +"them in your My Trusted Shops Account." +msgstr "" +"Exporter les données de vos clients et de vos commandes des x derniers jours " +"et importez-les dans votre compte My Trusted Shops." + +#: includes/class-wc-trusted-shops-admin.php:901 +msgctxt "trusted-shops" +msgid "30 days" +msgstr "30 jours" + +#: includes/class-wc-trusted-shops-admin.php:902 +msgctxt "trusted-shops" +msgid "60 days" +msgstr "60 jours" + +#: includes/class-wc-trusted-shops-admin.php:903 +msgctxt "trusted-shops" +msgid "90 days" +msgstr "90 jours" + +#: includes/class-wc-trusted-shops-admin.php:905 +#, php-format +msgctxt "trusted-shops" +msgid "Upload customer and order information %s." +msgstr "Cliquez %s pour importer les données de vos clients." + +#: includes/class-wc-trusted-shops-admin.php:908 +msgctxt "trusted-shops" +msgid "Days until reminder mail" +msgstr "Jours avant envoi du rappel" + +#: includes/class-wc-trusted-shops-admin.php:908 +msgctxt "trusted-shops" +msgid "" +"Set the number of days to wait after the order date before having a Review " +"Request sent to your customers." +msgstr "" +"Indiquer ici après combien de jours le rappel d’évaluation est envoyé à " +"l’acheteur après avoir atteint l’état de la commande sélectionné ci-dessus." + +#: includes/class-wc-trusted-shops-admin.php:912 +msgctxt "trusted-shops" +msgid "Start export" +msgstr "Débuter l’exportation" + +#: includes/class-wc-trusted-shops-review-exporter.php:64 +msgctxt "trusted-shops" +msgid "Order ID" +msgstr "Réf. commande" + +#: includes/class-wc-trusted-shops-review-exporter.php:65 +msgctxt "trusted-shops" +msgid "Order date" +msgstr "Date de la commande" + +#: includes/class-wc-trusted-shops-review-exporter.php:66 +msgctxt "trusted-shops" +msgid "# Days" +msgstr "# Jours" + +#: includes/class-wc-trusted-shops-review-exporter.php:67 +msgctxt "trusted-shops" +msgid "Email" +msgstr "Email" + +#: includes/class-wc-trusted-shops-review-exporter.php:68 +msgctxt "trusted-shops" +msgid "First name" +msgstr "Prénom" + +#: includes/class-wc-trusted-shops-review-exporter.php:69 +msgctxt "trusted-shops" +msgid "Last name" +msgstr "Nom" + +#: includes/class-wc-trusted-shops-template-hooks.php:121 +#, php-format +msgctxt "trusted-shops" +msgid "" +"Your review reminder e-mail has been cancelled successfully. Return to %s." +msgstr "" +"Votre e-mail de demande d’avis a été annulé avec succès. Retourner à %s." + +#: includes/class-wc-trusted-shops-template-hooks.php:121 +msgctxt "trusted-shops" +msgid "Home" +msgstr "Accueil" + +#: includes/class-wc-trusted-shops-template-hooks.php:193 +msgctxt "trusted-shops" +msgid "" +"Yes, I would like to be reminded via e-mail after {days} day(s) to review my " +"order. I am able to cancel the reminder at any time by clicking on the " +"\"cancel review reminder\" link within the order confirmation." +msgstr "" +"Oui, je souhaite recevoir un demande d’avis par e-mail au bout de {days} " +"jour(s) pour évaluer ma commande. Je peux annuler le rappel à tout moment en " +"cliquant sur le lien Annuler le rappel d’évaluation dans l’email de " +"confirmation de commande." + +#: includes/class-wc-trusted-shops-template-hooks.php:198 +msgctxt "trusted-shops" +msgid "Please allow us to send a review reminder by e-mail." +msgstr "" +"Veuillez nous autoriser à vous envoyer un rappel d’évaluation par e-mail." + +#: includes/class-wc-trusted-shops-template-hooks.php:201 +msgctxt "trusted-shops" +msgid "Review reminder" +msgstr "Demande d’avis" + +#: includes/class-wc-trusted-shops-template-hooks.php:202 +msgctxt "trusted-shops" +msgid "Asks the customer to receive a Trusted Shops review reminder." +msgstr "Demande au client s’il veut recevoir une demande d’avis Trusted Shops." + +#: includes/class-wc-ts-dependencies.php:36 +#: includes/class-wc-ts-dependencies.php:45 woocommerce-trusted-shops.php:67 +#: woocommerce-trusted-shops.php:76 +msgid "Cheatin’ huh?" +msgstr "Cheatin’ huh?" + +#: includes/class-wc-ts-install.php:72 +#, php-format +msgid "" +"Please install WooCommerce before " +"installing WooCommerce Germanized. Thank you!" +msgstr "" +"Please install WooCommerce before " +"installing WooCommerce Germanized. Thank you!" + +#: includes/class-wc-ts-settings-handler.php:23 +msgctxt "trusted-shops" +msgid "Trusted Shops" +msgstr "Trusted Shops" + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:25 +msgctxt "trusted-shops" +msgid "Trusted Shops Review Reminder" +msgstr " demande d’avis Trusted Shops" + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:26 +msgctxt "trusted-shops" +msgid "" +"This E-Mail is being sent to a customer to remind him about the possibility " +"to leave a review at Trusted Shops." +msgstr "" +"Cet e-mail est envoyé à un client pour lui rappeler qu’il peut laisser un " +"avis via Trusted Shops." + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:53 +msgctxt "trusted-shops" +msgid "Please rate your {site_title} order from {order_date}" +msgstr "" +"Veuillez évaluer votre commande sur la boutique {site_title} datant du " +"{order_date}" + +#: includes/emails/class-wc-ts-email-customer-trusted-shops.php:63 +msgctxt "trusted-shops" +msgid "Please rate your Order" +msgstr "Veuillez évaluer votre commande" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:19 +msgctxt "trusted-shops" +msgid "Show your TS shop review sticker." +msgstr "Afficher la vignette avis site" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:21 +msgctxt "trusted-shops" +msgid "Trusted Shops Shop Review Sticker" +msgstr "Vignette avis site" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:25 +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:46 +msgctxt "trusted-shops" +msgid "Trusted Shops Reviews" +msgstr "avis clients Trusted Shops" + +#: includes/widgets/class-wc-trusted-shops-widget-review-sticker.php:26 +msgctxt "trusted-shops" +msgid "Title" +msgstr "Titre" + +#: templates/emails/customer-trusted-shops.php:18 +#: templates/emails/plain/customer-trusted-shops.php:12 +#, php-format +msgctxt "trusted-shops" +msgid "Dear %s %s," +msgstr "Bonjour %s %s," + +#: templates/emails/customer-trusted-shops.php:19 +#: templates/emails/plain/customer-trusted-shops.php:14 +#, php-format +msgctxt "trusted-shops" +msgid "" +"You have recently shopped at %s. Thank you! We would be glad if you spent " +"some time to write a review about your order. To do so please follow follow " +"the link." +msgstr "" +"Vous avez récemment passé commande sur %s. Merci beaucoup! Nous serions " +"ravis si vous preniez quelques minutes afin de donner votre avis sur votre " +"commande. Veuillez pour cela cliquer sur le lien suivant." + +#: templates/emails/customer-trusted-shops.php:22 +msgctxt "trusted-shops" +msgid "Rate Order now" +msgstr "Évaluer ma commande maintenant" + +#: woocommerce-trusted-shops.php:220 +msgctxt "trusted-shops" +msgid "Yes" +msgstr "Oui" + +#: woocommerce-trusted-shops.php:220 +msgctxt "trusted-shops" +msgid "No" +msgstr "Non" + +#: woocommerce-trusted-shops.php:265 +#, php-format +msgctxt "trusted-shops" +msgid "" +"If the App helped you, please leave a %s★★" +"★★★%s in the Wordpress plugin repository." +msgstr "" +"If the App helped you, please leave a %s★★" +"★★★%s in the Wordpress plugin repository." + +#: woocommerce-trusted-shops.php:419 +msgctxt "trusted-shops" +msgid "Settings" +msgstr "Paramètres" diff --git a/packages/woocommerce-trusted-shops/includes/abstracts/abstract-wc-ts-compatibility.php b/packages/woocommerce-trusted-shops/includes/abstracts/abstract-wc-ts-compatibility.php new file mode 100644 index 000000000..d3b3731d8 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/abstracts/abstract-wc-ts-compatibility.php @@ -0,0 +1,87 @@ + '1.0.0', + 'requires_at_least' => '', + 'tested_up_to' => '', + ) + ); + + if ( empty( $version_data['requires_at_least'] ) && empty( $version_data['tested_up_to'] ) ) { + $version_data['requires_at_least'] = $version_data['version']; + $version_data['tested_up_to'] = $version_data['version']; + } elseif ( empty( $version_data['tested_up_to'] ) ) { + $version_data['tested_up_to'] = $version_data['requires_at_least']; + if ( WC_TS_Dependencies::instance()->compare_versions( $version_data['version'], $version_data['requires_at_least'], '>' ) ) { + $version_data['tested_up_to'] = $version_data['version']; + } + } elseif ( empty( $version_data['requires_at_least'] ) ) { + $version_data['requires_at_least'] = $version_data['tested_up_to']; + if ( WC_TS_Dependencies::instance()->compare_versions( $version_data['version'], $version_data['requires_at_least'], '<' ) ) { + $version_data['requires_at_least'] = $version_data['version']; + } + } + + $this->version_data = $version_data; + + $this->plugin_name = $plugin_name; + $this->plugin_file = $plugin_file; + + if ( ! $this->is_applicable() ) { + return; + } + + add_action( 'init', array( $this, 'early_execution' ), 0 ); + add_action( 'init', array( $this, 'load' ), 15 ); + + $this->after_plugins_loaded(); + } + + public function early_execution() {} + + public function after_plugins_loaded() {} + + public function is_applicable() { + return $this->is_activated() && $this->is_supported(); + } + + public function is_activated() { + return WC_TS_Dependencies::instance()->is_plugin_activated( $this->plugin_file ); + } + + public function is_supported() { + return WC_TS_Dependencies::instance()->compare_versions( $this->version_data['version'], $this->version_data['requires_at_least'], '>=' ) && + WC_TS_Dependencies::instance()->compare_versions( $this->version_data['version'], $this->version_data['tested_up_to'], '<=' ); + } + + public function get_name() { + return $this->plugin_name; + } + + public function get_version_data() { + return $this->version_data; + } + + abstract public function load(); +} diff --git a/packages/woocommerce-trusted-shops/includes/admin/settings/class-wc-ts-gzd-settings-tab.php b/packages/woocommerce-trusted-shops/includes/admin/settings/class-wc-ts-gzd-settings-tab.php new file mode 100644 index 000000000..e92f88de7 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/admin/settings/class-wc-ts-gzd-settings-tab.php @@ -0,0 +1,99 @@ +trusted_shops->get_dependency( 'admin' ); + + return $admin->get_trusted_url( 'https://support.trustedshops.com/en/apps/woocommerce' ); + } + + public function before_output() { + do_action( 'woocommerce_ts_admin_settings_before', $this->get_settings_for_section_core( '' ) ); + } + + public function review_exporter() { + $admin = WC_trusted_shops()->trusted_shops->get_dependency( 'admin' ); + $admin->review_collector_export(); + } + + public function register_scripts() { + if ( isset( $_GET['tab'] ) && 'germanized-trusted_shops' === $_GET['tab'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + do_action( 'woocommerce_trusted_shops_load_admin_scripts' ); + } + } + + public function get_description() { + return _x( 'Setup your Trusted Shops Integration.', 'trusted-shops', 'woocommerce-germanized' ); + } + + public function get_label() { + return _x( 'Trusted Shops', 'trusted-shops', 'woocommerce-germanized' ); + } + + public function get_name() { + return 'trusted_shops'; + } + + protected function output_description() {} + + public function get_tab_settings( $current_section = '' ) { + $admin = WC_trusted_shops()->trusted_shops->get_dependency( 'admin' ); + $settings = $admin->get_settings(); + + return $settings; + } + + public function get_sidebar( $current_section = '' ) { + $admin = WC_trusted_shops()->trusted_shops->get_dependency( 'admin' ); + $sidebar = $admin->get_sidebar(); + + return $sidebar; + } + + protected function get_enable_option_name() { + $option_prefix = Package::is_integration() ? 'gzd_' : ''; + $option_name = 'woocommerce_' . $option_prefix . 'trusted_shops_id'; + + return $option_name; + } + + public function is_enabled() { + $value = get_option( $this->get_enable_option_name() ); + + return ( ! empty( $value ) ? true : false ); + } + + protected function before_save( $settings, $current_section = '' ) { + do_action( 'woocommerce_ts_before_save', $settings ); + + parent::before_save( $settings, $current_section ); + } + + protected function after_save( $settings, $current_section = '' ) { + do_action( 'woocommerce_ts_after_save', $settings ); + + parent::after_save( $settings, $current_section ); + } +} diff --git a/packages/woocommerce-trusted-shops/includes/admin/views/html-notice-dependencies.php b/packages/woocommerce-trusted-shops/includes/admin/views/html-notice-dependencies.php new file mode 100644 index 000000000..059b5387e --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/admin/views/html-notice-dependencies.php @@ -0,0 +1,58 @@ + + +
+

+ plugins_required as $plugin => $data ) : // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited ?> + + is_plugin_activated( $plugin ) ) : + $missing_count++; + ?> + + + +

+ + + +

+ + + + is_plugin_outdated( $plugin ) ) : + $version_count ++; + ?> + +

+ + +

- ' . esc_html( $data['version'] ) . '' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>

+ + + + + + 0 ) : ?> + +

+ + + +

+ + + +
diff --git a/packages/woocommerce-trusted-shops/includes/admin/views/html-notice-update.php b/packages/woocommerce-trusted-shops/includes/admin/views/html-notice-update.php new file mode 100644 index 000000000..2d2b4923b --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/admin/views/html-notice-update.php @@ -0,0 +1,20 @@ + +
+

WooCommerce Trusted Shops Data Update Required – We just need to update your installation to the latest version', 'trusted-shops', 'woocommerce-germanized' ) ); ?>

+

+
+ diff --git a/packages/woocommerce-trusted-shops/includes/admin/views/html-settings-section.php b/packages/woocommerce-trusted-shops/includes/admin/views/html-settings-section.php new file mode 100644 index 000000000..f89cf2358 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/admin/views/html-settings-section.php @@ -0,0 +1,44 @@ + +
+
+ + + +
+ + +
+ +
+ +
diff --git a/packages/woocommerce-trusted-shops/includes/admin/views/html-wpml-notice.php b/packages/woocommerce-trusted-shops/includes/admin/views/html-wpml-notice.php new file mode 100644 index 000000000..125338467 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/admin/views/html-wpml-notice.php @@ -0,0 +1,20 @@ + + +
+

+

+ + + + ' . $current_language . '' ) ); ?> + +

+
diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-admin.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-admin.php new file mode 100644 index 000000000..9e8d21ed8 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-admin.php @@ -0,0 +1,1020 @@ +base = $base; + + add_action( 'woocommerce_ts_before_save', array( $this, 'before_save' ), 0, 1 ); + add_action( 'woocommerce_ts_after_save', array( $this, 'after_save' ), 0, 1 ); + + add_action( 'woocommerce_ts_admin_settings_before', array( $this, 'wpml_notice' ) ); + + // Default settings + add_filter( 'woocommerce_gzd_installation_default_settings', array( $this, 'set_installation_settings' ), 10, 1 ); + + // After Install + add_action( 'woocommerce_trusted_shops_installed', array( $this, 'create_attribute' ) ); + + // Review Collector + add_action( 'admin_init', array( $this, 'review_collector_export_csv' ) ); + add_action( 'woocommerce_trusted_shops_load_admin_scripts', array( $this, 'load_scripts' ) ); + + if ( ! \Vendidero\TrustedShops\Package::is_integration() ) { + add_action( 'woocommerce_product_options_general_product_data', array( $this, 'output_fields' ) ); + add_action( 'woocommerce_product_after_variable_attributes', array( $this, 'output_variation_fields' ), 20, 3 ); + add_action( 'woocommerce_save_product_variation', array( $this, 'save_variation_fields' ), 0, 2 ); + + if ( ! wc_ts_woocommerce_supports_crud() ) { + add_action( 'woocommerce_process_product_meta', array( $this, 'save_fields' ), 20, 2 ); + } else { + add_action( 'woocommerce_admin_process_product_object', array( $this, 'save_fields' ), 10, 1 ); + } + } + } + + public function set_installation_settings( $settings ) { + return array_merge( $settings, $this->get_settings() ); + } + + public function wpml_notice() { + if ( $this->base->is_multi_language_setup() ) { + $is_default_language = false; + $compatibility = $this->base->get_multi_language_compatibility(); + $default_language = strtoupper( $compatibility->get_default_language() ); + $current_language = strtoupper( $compatibility->get_current_language() ); + + if ( $current_language === $default_language ) { + $is_default_language = true; + } + + include_once 'admin/views/html-wpml-notice.php'; + } + } + + public function output_variation_fields( $loop, $variation_data, $variation ) { + $_product = wc_get_product( $variation ); + $_parent = wc_get_product( wc_ts_get_crud_data( $_product, 'parent' ) ); + $variation_id = wc_ts_get_crud_data( $_product, 'id' ); + $variation_meta = get_post_meta( $variation_id ); + $variation_data = array(); + + $variation_fields = array( + '_ts_gtin' => '', + '_ts_mpn' => '', + ); + + foreach ( $variation_fields as $field => $value ) { + $variation_data[ $field ] = isset( $variation_meta[ $field ][0] ) ? maybe_unserialize( $variation_meta[ $field ][0] ) : $value; + } + + ?> + +
+

+ + +

+

+ + +

+
+ + '', + '_ts_mpn' => '', + ); + + foreach ( $data as $k => $v ) { + $data_k = 'variable' . ( substr( $k, 0, 1 ) === '_' ? '' : '_' ) . $k; + $data[ $k ] = ( isset( $_POST[ $data_k ][ $i ] ) ? wc_clean( wp_unslash( $_POST[ $data_k ][ $i ] ) ) : null ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( $product = wc_get_product( $variation_id ) ) { + foreach ( $data as $key => $value ) { + $product = wc_ts_set_crud_data( $product, $key, $value ); + } + + if ( wc_ts_woocommerce_supports_crud() ) { + $product->save(); + } + } + } + + public function output_fields() { + echo '
'; + + woocommerce_wp_text_input( + array( + 'id' => '_ts_gtin', + 'label' => _x( 'GTIN', 'trusted-shops', 'woocommerce-germanized' ), + 'data_type' => 'text', + 'desc_tip' => true, + 'description' => _x( + 'ID that allows your products to be identified worldwide. If you want to display your Trusted Shops Product Reviews in Google Shopping and paid Google adverts, Google needs the GTIN.', + 'trusted-shops', + 'woocommerce-trusted-shops' + ), + ) + ); + woocommerce_wp_text_input( + array( + 'id' => '_ts_mpn', + 'label' => _x( 'MPN', 'trusted-shops', 'woocommerce-germanized' ), + 'data_type' => 'text', + 'desc_tip' => true, + 'description' => _x( + 'If you don\'t have a GTIN for your products, you can pass the brand name and the MPN on to Google to use the Trusted Shops Google Integration.', + 'trusted-shops', + 'woocommerce-trusted-shops' + ), + ) + ); + + echo '
'; + } + + public function save_fields( $product ) { + if ( is_numeric( $product ) ) { + $product = wc_get_product( $product ); + } + + if ( isset( $_POST['_ts_gtin'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $product = wc_ts_set_crud_data( $product, '_ts_gtin', wc_clean( wp_unslash( $_POST['_ts_gtin'] ) ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + + if ( isset( $_POST['_ts_mpn'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + $product = wc_ts_set_crud_data( $product, '_ts_mpn', wc_clean( wp_unslash( $_POST['_ts_mpn'] ) ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing + } + } + + public function create_attribute() { + $attributes = array( + 'brand' => _x( 'Brand', 'trusted-shops', 'woocommerce-germanized' ), + ); + + // Create the taxonomy + global $wpdb; + delete_transient( 'wc_attribute_taxonomies' ); + + foreach ( $attributes as $attribute_name => $title ) { + if ( ! in_array( 'pa_' . $attribute_name, wc_get_attribute_taxonomy_names(), true ) ) { + $attribute = array( + 'attribute_label' => $title, + 'attribute_name' => $attribute_name, + 'attribute_type' => 'text', + 'attribute_orderby' => 'menu_order', + 'attribute_public' => 0, + ); + + $wpdb->insert( $wpdb->prefix . 'woocommerce_attribute_taxonomies', $attribute ); + delete_transient( 'wc_attribute_taxonomies' ); + } + } + } + + public function load_scripts() { + $screen = get_current_screen(); + $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; + $assets_path = $this->base->plugin->plugin_url() . '/assets/'; + + wp_register_style( 'woocommerce-trusted-shops-admin', $assets_path . 'css/admin' . $suffix . '.css', false, $this->base->plugin->version ); + wp_enqueue_style( 'woocommerce-trusted-shops-admin' ); + + wp_register_script( 'wc-admin-trusted-shops', $assets_path . 'js/admin' . $suffix . '.js', array( 'jquery', 'woocommerce_settings' ), $this->base->plugin->version, true ); + wp_localize_script( + 'wc-admin-trusted-shops', + 'trusted_shops_params', + array( + 'option_prefix' => $this->base->option_prefix, + 'i18n_error_mandatory' => _x( 'This field is mandatory', 'trusted-shops', 'woocommerce-germanized' ), + ) + ); + + wp_enqueue_script( 'wc-admin-trusted-shops' ); + } + + public function register_section( $sections ) { + $sections['trusted_shops'] = _x( 'Trusted Shops Options', 'trusted-shops', 'woocommerce-germanized' ); + + return $sections; + } + + public function get_font_families() { + return array( + 'Arial' => _x( 'Arial', 'trusted-shops', 'woocommerce-germanized' ), + 'Geneva' => _x( 'Geneva', 'trusted-shops', 'woocommerce-germanized' ), + 'Georgia' => _x( 'Georgia', 'trusted-shops', 'woocommerce-germanized' ), + 'Helvetica' => _x( 'Helvetica', 'trusted-shops', 'woocommerce-germanized' ), + 'Sans-serif' => _x( 'Sans-serif', 'trusted-shops', 'woocommerce-germanized' ), + 'Serif' => _x( 'Serif', 'trusted-shops', 'woocommerce-germanized' ), + 'Trebuchet MS' => _x( 'Trebuchet MS', 'trusted-shops', 'woocommerce-germanized' ), + 'Verdana' => _x( 'Verdana', 'trusted-shops', 'woocommerce-germanized' ), + ); + } + + public function get_order_statuses() { + return wc_get_order_statuses(); + } + + public function get_translatable_settings() { + $translatable = array(); + + foreach ( $this->get_settings_array() as $setting ) { + if ( isset( $setting['id'] ) && ! in_array( $setting['type'], array( 'title', 'sectionend' ), true ) ) { + $translatable[ $setting['id'] ] = ''; + } + } + + return $translatable; + } + + protected function get_settings_array( $defaults = array() ) { + $settings = array( + array( + 'title' => _x( 'Trusted Shops Integration', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => sprintf( _x( 'Do you need help with integrating your Trustbadge? %s', 'trusted-shops', 'woocommerce-germanized' ), '' . _x( 'To the step-by-step instructions', 'trusted-shops', 'woocommerce-germanized' ) . '' ), + 'type' => 'title', + 'id' => 'trusted_shops_options', + ), + + array( + 'title' => _x( 'Trusted Shops ID', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( 'The Trusted Shops ID is a unique identifier for your shop. You can find your Trusted Shops ID in your My Trusted Shops account.', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => true, + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_id', + 'type' => 'text', + 'custom_attributes' => array( 'data-sidebar' => 'wc-ts-sidebar-default' ), + 'css' => 'min-width:300px;', + ), + + array( + 'title' => _x( 'Edit Mode', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_integration_mode', + 'desc_tip' => _x( 'The advanced configuration is for users with programming skills. Here you can create even more individual settings.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'select', + 'class' => 'chosen_select', + 'options' => array( + 'standard' => _x( 'Standard configuration', 'trusted-shops', 'woocommerce-germanized' ), + 'expert' => _x( 'Advanced configuration', 'trusted-shops', 'woocommerce-germanized' ), + ), + 'default' => 'standard', + ), + + array( + 'type' => 'sectionend', + 'id' => 'trusted_shops_options', + ), + + array( + 'title' => _x( 'Configure your Trustbadge', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'trusted_shops_badge_options', + ), + + array( + 'title' => _x( 'Display Trustbadge', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_trustbadge_enable', + 'desc_tip' => _x( 'Display the Trustbadge on all the pages of your shop.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'ts_toggle', + 'custom_attributes' => array( 'data-sidebar' => 'wc-ts-sidebar-trustbadge' ), + 'default' => 'no', + ), + + array( + 'title' => _x( 'Variant', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_trustbadge_variant', + 'desc_tip' => _x( 'You can display your Trustbadge with or without Review Stars.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'select', + 'class' => 'chosen_select', + 'options' => array( + 'standard' => _x( 'Display Trustbadge with review stars', 'trusted-shops', 'woocommerce-germanized' ), + 'hide_reviews' => _x( 'Display Trustbadge without review stars', 'trusted-shops', 'woocommerce-germanized' ), + ), + 'default' => 'standard', + ), + + array( + 'title' => _x( 'Vertical Offset', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Choose the distance that the Trustbadge will appear from the bottom-right corner of the screen.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_trustbadge_y', + 'type' => 'number', + 'desc' => _x( 'px', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => '0', + 'custom_attributes' => array( + 'step' => '1', + 'min' => 0, + 'data-validate' => 'integer', + 'data-validate-msg' => sprintf( _x( 'Please choose a non-negative number (at least %d)', 'trusted-shops', 'woocommerce-germanized' ), 0 ), + ), + 'css' => 'max-width:60px;', + ), + + array( + 'title' => _x( 'Trustbadge code', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_trustbadge_code', + 'type' => 'textarea', + 'desc_tip' => true, + 'desc' => _x( 'The advanced configuration is for users with programming skills. Here you can create even more individual settings.', 'trusted-shops', 'woocommerce-germanized' ), + 'css' => 'width: 100%; min-height: 150px', + 'default' => '', + ), + + array( + 'type' => 'sectionend', + 'id' => 'trusted_shops_badge_options', + ), + + array( + 'title' => _x( 'Configure your Shop Reviews', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'trusted_shops_review_sticker_options', + ), + + array( + 'title' => _x( 'Display Shop Review Sticker', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'To display the Shop Review Sticker, you have to assign the widget "Trusted Shops Review Sticker".', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => sprintf( _x( 'Assign widget %s', 'trusted-shops', 'woocommerce-germanized' ), '' . _x( 'here', 'trusted-shops', 'woocommerce-germanized' ) . '' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_enable', + 'type' => 'ts_toggle', + 'custom_attributes' => array( 'data-sidebar' => 'wc-ts-sidebar-shop-reviews' ), + 'default' => 'no', + ), + + array( + 'title' => _x( 'Background color', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Choose the background color for your Review Sticker.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_bg_color', + 'type' => 'color', + 'default' => '#FFDC0F', + ), + + array( + 'title' => _x( 'Font', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_font', + 'desc_tip' => _x( 'Choose the font for your Review Sticker.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'select', + 'class' => 'chosen_select', + 'default' => 'arial', + ), + + array( + 'title' => _x( 'Number of reviews displayed', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Display x alternating Shop Reviews in your Shop Review Sticker. You can display between 1 and 5 alternating Shop Reviews.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_number', + 'type' => 'number', + 'desc' => _x( 'Show x alternating reviews', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => '5', + 'custom_attributes' => array( + 'step' => '1', + 'min' => 1, + 'max' => 5, + 'data-validate' => 'integer', + 'data-validate-msg' => sprintf( _x( 'Please choose a non-negative number between %1$d and %2$d', 'trusted-shops', 'woocommerce-germanized' ), 1, 5 ), + ), + 'css' => 'max-width:60px;', + ), + + array( + 'title' => _x( 'Minimum rating displayed', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Only show Shop Reviews with a minimum rating of x stars. ', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_better_than', + 'type' => 'number', + 'desc' => _x( 'Star(s)', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => '3', + 'custom_attributes' => array( + 'step' => '1', + 'min' => 1, + 'max' => 5, + 'data-validate' => 'integer', + 'data-validate-msg' => sprintf( _x( 'Please choose a non-negative number between %1$d and %2$d', 'trusted-shops', 'woocommerce-germanized' ), 1, 5 ), + ), + 'css' => 'max-width:60px;', + ), + + array( + 'title' => _x( 'Sticker code', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_code', + 'type' => 'textarea', + 'desc_tip' => true, + 'desc' => _x( 'The advanced configuration is for users with programming skills. Here you can perform even more individual settings.', 'trusted-shops', 'woocommerce-germanized' ), + 'css' => 'width: 100%; min-height: 150px', + 'default' => '', + ), + + array( + 'title' => _x( 'Google Organic Search', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Activate this option to give Google the opportunity to show your Shop Reviews in Google organic search results.', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( 'By activating this option, rich snippets will be integrated in the selected pages so your shop review stars may be displayed in Google organic search results. If you use Product Reviews and already activated rich snippets in expert mode, we recommend integrating rich snippets for Shop Reviews on category pages only.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_rich_snippets_enable', + 'type' => 'ts_toggle', + 'default' => 'no', + ), + + array( + 'title' => _x( 'Activate rich snippets on', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( 'category pages', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_rich_snippets_category', + 'type' => 'checkbox', + 'default' => 'no', + 'checkboxgroup' => 'start', + ), + + array( + 'desc' => _x( 'product pages', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_rich_snippets_product', + 'type' => 'checkbox', + 'default' => 'no', + 'checkboxgroup' => '', + ), + + array( + 'desc' => _x( 'homepage (not recommended)', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_rich_snippets_home', + 'type' => 'checkbox', + 'default' => 'no', + 'checkboxgroup' => 'end', + ), + + array( + 'title' => _x( 'Rich snippets code', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_rich_snippets_code', + 'type' => 'textarea', + 'desc_tip' => true, + 'desc' => _x( 'The advanced configuration is for users with programming skills. Here you can create even more individual settings.', 'trusted-shops', 'woocommerce-germanized' ), + 'css' => 'width: 100%; min-height: 150px', + 'default' => '', + ), + + array( + 'type' => 'sectionend', + 'id' => 'trusted_shops_review_sticker_options', + ), + + array( + 'title' => _x( 'Configure your Product Reviews ', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => sprintf( _x( 'To use Product Reviews, activate them in your %s first.', 'trusted-shops', 'woocommerce-germanized' ), '' . _x( 'Trusted Shops package', 'trusted-shops', 'woocommerce-germanized' ) . '' ), + 'type' => 'title', + 'id' => 'trusted_shops_reviews_options', + ), + + array( + 'title' => _x( 'Collect Product Reviews', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( 'Show Product Reviews on the product page in a separate tab, just as shown on the picture on the right.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_reviews_enable', + 'type' => 'ts_toggle', + 'custom_attributes' => array( 'data-sidebar' => 'wc-ts-sidebar-product-reviews' ), + 'default' => 'no', + ), + + array( + 'title' => _x( 'Reviews', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'You can choose a name for the tab with your Product Reviews.', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( 'Show Product Reviews on the product detail page in an additional tab.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_enable', + 'type' => 'ts_toggle', + 'default' => 'no', + ), + + array( + 'title' => _x( 'Name of Product Reviews tab', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'You can choose a name for the tab with your Product Reviews.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_tab_text', + 'type' => 'text', + 'default' => _x( 'Product reviews', 'trusted-shops', 'woocommerce-germanized' ), + ), + + array( + 'title' => _x( 'Border color', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Set the color for the frame around your Product Reviews.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_border_color', + 'type' => 'color', + 'default' => '#FFDC0F', + ), + + array( + 'title' => _x( 'Background color', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Set the background color for your Product Reviews.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_bg_color', + 'type' => 'color', + 'default' => '#FFFFFF', + ), + + array( + 'title' => _x( 'Star color', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Set the color for the Product Review stars in your Product Reviews tab.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_star_color', + 'type' => 'color', + 'default' => '#C0C0C0', + ), + + array( + 'title' => _x( 'Star size', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_star_size', + 'type' => 'number', + 'default' => '15', + 'desc' => _x( 'px', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Set the size for the Product Review stars in your Product Reviews tab.', 'trusted-shops', 'woocommerce-germanized' ), + 'css' => 'max-width:60px;', + 'custom_attributes' => array( + 'step' => '1', + 'min' => 0, + 'data-validate' => 'integer', + 'data-validate-msg' => sprintf( _x( 'Please choose a non-negative number (at least %d)', 'trusted-shops', 'woocommerce-germanized' ), 0 ), + ), + ), + + array( + 'title' => _x( 'Product Sticker Code', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_code', + 'desc_tip' => _x( 'The advanced configuration is for users with programming skills. Here you can perform even more individual settings.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'textarea', + 'css' => 'width: 100%; min-height: 150px', + 'default' => '', + ), + + array( + 'title' => _x( 'jQuerySelector', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Please choose where your Product Reviews shall be displayed on the Product detail page.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_selector', + 'type' => 'text', + 'default' => '#ts_product_sticker', + ), + + array( + 'title' => _x( 'Rating stars', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( 'Show star ratings on the product detail page below your product name.', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Display Product Review stars on product pages below the product name, just as shown in the picture on the right.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_enable', + 'type' => 'ts_toggle', + 'default' => 'no', + ), + + array( + 'title' => _x( 'Star color', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_star_color', + 'desc_tip' => _x( 'Set the color for the review stars, that are displayed on the product page, below your product name.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'color', + 'default' => '#FFDC0F', + ), + + array( + 'title' => _x( 'Star size', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_star_size', + 'desc_tip' => _x( 'Set the size for the review stars that are displayed on the product page, below your product name.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'number', + 'default' => '14', + 'desc' => _x( 'px', 'trusted-shops', 'woocommerce-germanized' ), + 'css' => 'max-width:60px;', + 'custom_attributes' => array( + 'step' => '1', + 'min' => 0, + 'data-validate' => 'integer', + 'data-validate-msg' => sprintf( _x( 'Please choose a non-negative number (at least %d)', 'trusted-shops', 'woocommerce-germanized' ), 0 ), + ), + ), + + array( + 'title' => _x( 'Font size', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_font_size', + 'desc_tip' => _x( 'Set the font size for the text that goes with your review stars.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'number', + 'desc' => _x( 'px', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => '12', + 'css' => 'max-width:60px;', + 'custom_attributes' => array( + 'step' => '1', + 'min' => 0, + 'data-validate' => 'integer', + 'data-validate-msg' => sprintf( _x( 'Please choose a non-negative number (at least %d)', 'trusted-shops', 'woocommerce-germanized' ), 0 ), + ), + ), + + array( + 'title' => _x( 'Product Review Code', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'The advanced configuration is for users with programming skills. Here you can perform even more individual settings.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_code', + 'type' => 'textarea', + 'css' => 'width: 100%; min-height: 150px', + 'default' => '', + ), + + array( + 'title' => _x( 'jQuerySelector', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Please choose where your Product Review Stars shall be displayed on the Product Detail page.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_selector', + 'type' => 'text', + 'default' => '#ts_product_widget', + ), + + array( + 'title' => _x( 'Brand attribute', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => sprintf( _x( 'Create brand attribute %s', 'trusted-shops', 'woocommerce-germanized' ), '' . _x( 'here', 'trusted-shops', 'woocommerce-germanized' ) . '' ), + 'desc_tip' => _x( 'Brand name of the product. By passing this information on to Google, you improve your chances of having Google identify your products. Assign your brand attribute. If your products don\'t have a GTIN, you can pass on the brand name and the MPN to use Google Integration.', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_brand_attribute', + 'css' => 'min-width:250px;', + 'default' => 'brand', + 'type' => 'select', + 'class' => 'chosen_select_nostd', + 'custom_attributes' => array( 'data-placeholder' => _x( 'None', 'trusted-shops', 'woocommerce-germanized' ) ), + ), + + array( + 'type' => 'sectionend', + 'id' => 'trusted_shops_reviews_options', + ), + ); + + if ( $this->base->supports( 'reminder' ) ) { + + $settings = array_merge( + $settings, + array( + + array( + 'title' => _x( 'Configure your Review Requests', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( '7 days after an order has been placed, Trusted Shops automatically sends an invite to your customers. If you want to set a different time for sending automatic Review Requests, please activate the option below. If you want to send review requests with legal certainty, you need your customers\' consent to receive Review Requests. You also have to include an option to unsubscribe.', 'trusted-shops', 'woocommerce-germanized' ), + 'type' => 'title', + 'id' => 'trusted_shops_review_reminder_options', + ), + + array( + 'title' => _x( 'Enable Review Requests', 'trusted-shops', 'woocommerce-germanized' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_reminder_enable', + 'type' => 'ts_toggle', + 'default' => 'no', + 'custom_attributes' => array( 'data-sidebar' => 'wc-ts-sidebar-review-reminder' ), + 'autoload' => false, + ), + + array( + 'title' => _x( 'WooCommerce status', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'We recommend choosing the order status that you set when your products have been shipped.', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => array( 'wc-completed' ), + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_reminder_status', + 'type' => 'multiselect', + 'class' => 'chosen_select', + ), + + array( + 'title' => _x( 'Days until Review Request', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'Set the number of days to wait after an order has reached the order status you selected above before having a review request sent to your customers.', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => 7, + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_reminder_days', + 'type' => 'number', + 'custom_attributes' => array( + 'step' => '1', + 'min' => 0, + 'data-validate' => 'integer', + ), + ), + + array( + 'title' => _x( 'Permission via checkbox', 'trusted-shops', 'woocommerce-germanized' ), + 'desc_tip' => _x( 'If the checkbox is activated, only customers who gave their consent will receive Review Requests.', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => '', + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_reminder_checkbox', + 'type' => 'html', + 'html' => '' . _x( 'Edit checkbox', 'trusted-shops', 'woocommerce-germanized' ) . '', + ), + + array( + 'title' => _x( 'Unsubscribe via link', 'trusted-shops', 'woocommerce-germanized' ), + 'desc' => _x( 'Allows the customer to unsubscribe from Review Requests.', 'trusted-shops', 'woocommerce-germanized' ), + 'default' => 'yes', + 'id' => 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_reminder_opt_out', + 'type' => 'ts_toggle', + ), + + array( + 'type' => 'sectionend', + 'id' => 'trusted_shops_review_reminder_options', + ), + + ) + ); + } + + if ( ! empty( $defaults ) ) { + foreach ( $settings as $key => $setting ) { + if ( isset( $setting['id'] ) ) { + foreach ( $defaults as $setting_id => $default ) { + if ( $setting_id === $setting['id'] ) { + $settings[ $key ] = array_replace_recursive( $setting, $default ); + } + } + } + } + } + + return $settings; + } + + /** + * Get Trusted Shops related Settings for Admin Interface + * + * @return array + */ + public function get_settings() { + + $attributes = wc_get_attribute_taxonomies(); + $linked_attributes = array(); + + // Set attributes + foreach ( $attributes as $attribute ) { + $linked_attributes[ $attribute->attribute_name ] = $attribute->attribute_label; + } + + // Add empty option placeholder to allow clearing + $linked_attributes = array_merge( array( '' => '' ), $linked_attributes ); + + $update_settings = array( + 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_trustbadge_code' => array( + 'default' => $this->base->get_trustbadge_code( false ), + ), + 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_font' => array( + 'options' => $this->get_font_families(), + ), + 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_code' => array( + 'default' => $this->base->get_review_sticker_code( false ), + ), + 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_rich_snippets_code' => array( + 'default' => $this->base->get_rich_snippets_code( false ), + ), + 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_code' => array( + 'default' => $this->base->get_product_sticker_code( false ), + ), + 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_code' => array( + 'default' => $this->base->get_product_widget_code( false ), + ), + 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_brand_attribute' => array( + 'options' => $linked_attributes, + ), + ); + + if ( $this->base->supports( 'reminder' ) ) { + $update_settings[ 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_reminder_status' ] = array( + 'options' => $this->get_order_statuses(), + ); + } + + $settings = $this->get_settings_array( $update_settings ); + + return $settings; + } + + public function get_image( $img ) { + $language = $this->base->get_language(); + $endings = array( '.jpg', '.png' ); + $last = substr( $img, -4 ); + $ending = ''; + + if ( in_array( $last, $endings, true ) ) { + $ending = $last; + $img = substr( $img, 0, -4 ); + } + + $new_img = $img . '_' . $language . $ending; + + return $this->base->plugin->plugin_url() . '/assets/images/ts/' . $new_img; + } + + public function get_sidebar() { + ob_start(); + ?> +
+
+
+

+ + +
+ +
+
+ + +
+
+ + +
+
+ +
+ +
+ +
+ + + + + +
+ +
+

Settings and deactivate "Collect reviews automatically"', 'trusted-shops', 'woocommerce-germanized' ); ?>

+ +
+ +
+

Shop Reviews > Review Collector', 'trusted-shops', 'woocommerce-germanized' ); ?>

+ +
+
+
+ base->option_prefix . 'trusted_shops_id' ] ) && $_POST[ 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_id' ] !== $this->base->id ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + update_option( '_woocommerce_' . $this->base->option_prefix . 'trusted_shops_update_reviews', 1 ); + } + } + + public function after_save( $settings ) { + $this->base->refresh(); + + if ( get_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_integration_mode' ) === 'standard' ) { + // Delete code snippets + delete_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_trustbadge_code' ); + delete_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_code' ); + delete_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_code' ); + delete_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_rich_snippets_code' ); + delete_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_sticker_code' ); + update_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_widget_selector', '#ts_product_widget' ); + update_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_product_sticker_selector', '#ts_product_sticker' ); + } + + // Disable Reviews if Trusted Shops review collection has been enabled + if ( get_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_reviews_enable' ) === 'yes' ) { + update_option( 'woocommerce_enable_review_rating', 'no' ); + } + + if ( get_option( '_woocommerce_' . $this->base->option_prefix . 'trusted_shops_update_reviews' ) ) { + $this->base->get_dependency( 'schedule' )->update_reviews(); + } + + delete_option( '_woocommerce_' . $this->base->option_prefix . 'trusted_shops_update_reviews' ); + } + + public function review_collector_export_csv() { + if ( ! current_user_can( 'manage_woocommerce' ) ) { + return; + } + + if ( ! isset( $_GET['action'] ) || 'wc_' . $this->base->option_prefix . 'trusted-shops-export' !== $_GET['action'] || ( isset( $_GET['action'], $_REQUEST['_wpnonce'] ) && 'wc_' . $this->base->option_prefix . 'trusted-shops-export' === $_GET['action'] && ! wp_verify_nonce( wp_unslash( $_REQUEST['_wpnonce'] ), 'wc_' . $this->base->option_prefix . 'trusted-shops-export' ) ) ) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized + return; + } + + $interval_d = ( ( isset( $_GET['interval'] ) && ! empty( $_GET['interval'] ) ) ? absint( $_GET['interval'] ) : 30 ); + $days_to_send = ( ( isset( $_GET['days'] ) && ! empty( $_GET['days'] ) ) ? absint( $_GET['days'] ) : 5 ); + $status = ( ( isset( $_GET['status'] ) && ! empty( $_GET['status'] ) ) ? wc_clean( wp_unslash( $_GET['status'] ) ) : '' ); + + update_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_collector_days_to_send', $days_to_send ); + update_option( 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_review_collector', $interval_d ); + + if ( wc_ts_woocommerce_supports_crud() ) { + include_once 'class-wc-trusted-shops-review-exporter.php'; + + $exporter = new WC_Trusted_Shops_Review_Exporter(); + $exporter->set_days_until_send( $days_to_send ); + $exporter->set_interval_days( $interval_d ); + + if ( ! empty( $status ) ) { + $exporter->set_statuses( array( $status ) ); + } + + if ( isset( $_GET['lang'] ) && ! empty( $_GET['lang'] ) ) { + $exporter->set_lang( wc_clean( wp_unslash( $_GET['lang'] ) ) ); + } + + $exporter->export(); + } + } + + public function review_collector_export() { + + $href_org = admin_url(); + $href_org = add_query_arg( + array( + 'action' => 'wc_' . $this->base->option_prefix . 'trusted-shops-export', + '_wpnonce' => wp_create_nonce( 'wc_' . $this->base->option_prefix . 'trusted-shops-export' ), + 'lang' => $this->base->is_multi_language_setup() ? $this->base->get_multi_language_compatibility()->get_current_language() : '', + ), + $href_org + ); + + if ( ! wc_ts_woocommerce_supports_crud() ) { + return; + } + + $days_interval = get_option( 'woocommerce_gzd_trusted_shops_review_collector', 30 ); + $days_to_send = get_option( 'woocommerce_gzd_trusted_shops_review_collector_days_to_send', 5 ); + ?> +

+
+

get_trusted_url( 'https://www.trustedshops.com/tsb2b/sa/ratings/reviewCollector/reviewCollector.seam' ) ) . '" target="_blank">' . esc_html_x( 'My Trusted Shops account', 'trusted-shops', 'woocommerce-germanized' ) . '' ) ); ?>

+
+ + + + + + + +
+ + + + get_trusted_url( 'https://www.trustedshops.com/tsb2b/sa/ratings/reviewCollector/reviewCollector.seam' ) ) . '" target="_blank">' . esc_html_x( 'here', 'trusted-shops', 'woocommerce-germanized' ) . '' ) ); ?> +
+
+ + +
+
+ +
+
+
+ false, + ) + ); + + $url = empty( $url ) ? $this->base->signup_url : $url; + + return $this->get_trusted_url( $url, $args ); + } + + public function get_trusted_url( $url, $args = array() ) { + $param_args = $this->base->et_params; + $args = wp_parse_args( + $args, + array( + 'utm_term' => substr( get_locale(), 0, 2 ), + 'shop_id' => $this->base->ID, + 'params' => false, + 'lang_mapping' => array(), + ) + ); + + $current_lang = $this->base->get_language(); + + $base_lang = isset( $args['lang_mapping']['en'] ) ? $args['lang_mapping']['en'] : 'en'; + $current_lang = isset( $args['lang_mapping'][ $current_lang ] ) ? $args['lang_mapping'][ $current_lang ] : $current_lang; + $url = str_replace( "/{$base_lang}/", '/' . $current_lang . '/', $url ); + + if ( 'gzd_' === $this->base->option_prefix && substr( $url, -11 ) === 'woocommerce' ) { + $url = str_replace( 'woocommerce', 'woocommerce_germanized', $url ); + } + + if ( $args['params'] ) { + $param_args = array_replace_recursive( + $param_args, + array( + 'utm_term' => $args['utm_term'], + 'shop_id' => $args['shop_id'], + ) + ); + + return esc_url_raw( add_query_arg( $param_args, $url ) ); + } else { + return $url; + } + } + +} diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-core.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-core.php new file mode 100644 index 000000000..82b8e209e --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-core.php @@ -0,0 +1,471 @@ +version = Package::get_version(); + + // Define constants + $this->define_constants(); + + // Include required files + $this->includes(); + $this->setup_compatibility(); + + // Hooks + add_filter( 'plugin_action_links_' . plugin_basename( WC_TRUSTED_SHOPS_PLUGIN_FILE ), array( $this, 'action_links' ) ); + add_action( 'init', array( $this, 'init' ), 1 ); + add_filter( 'woocommerce_locate_template', array( $this, 'filter_templates' ), 0, 3 ); + + // Initialize Trusted Shops module + $this->trusted_shops = new WC_Trusted_Shops( + $this, + array( + 'supports' => Package::is_integration() ? array( 'reminder' ) : array(), + 'prefix' => Package::is_integration() ? 'GZD_' : '', + 'signup_url' => 'http://www.trustbadge.com/de/Preise/', + 'path' => WC_TRUSTED_SHOPS_ABSPATH . 'includes/', + ) + ); + + // Loaded action + do_action( 'woocommerce_trusted_shops_loaded' ); + } + + /** + * Init Trusted Shops when WordPress initializes. + */ + public function init() { + // Before init action + do_action( 'before_woocommerce_trusted_shops_init' ); + + if ( ! Package::is_integration() ) { + $this->load_plugin_textdomain(); + add_filter( 'woocommerce_get_settings_pages', array( $this, 'add_settings' ) ); + } else { + add_filter( 'woocommerce_email_classes', array( $this, 'add_emails' ), 10 ); + add_filter( 'woocommerce_gzd_wpml_email_ids', array( $this, 'add_wpml_emails' ), 10 ); + add_filter( 'woocommerce_gzd_admin_settings_tabs', array( $this, 'add_germanized_settings_tab' ), 10, 1 ); + } + + add_action( 'admin_enqueue_scripts', array( $this, 'add_admin_styles' ) ); + add_filter( 'admin_footer_text', array( $this, 'admin_footer_text' ), 15 ); + add_action( 'admin_print_styles', array( $this, 'add_notices' ), 1 ); + + // Change email template path if is germanized email template + add_filter( 'woocommerce_template_directory', array( $this, 'set_woocommerce_template_dir' ), 10, 2 ); + add_filter( 'woocommerce_locate_core_template', array( $this, 'set_woocommerce_core_template_dir' ), 10, 3 ); + + add_action( 'woocommerce_admin_field_ts_toggle', array( $this, 'toggle_input' ), 10 ); + add_filter( 'woocommerce_admin_settings_sanitize_option', array( $this, 'save_toggle_input_field' ), 0, 3 ); + + // Init action + do_action( 'woocommerce_trusted_shops_init' ); + } + + public function add_germanized_settings_tab( $tabs ) { + include_once dirname( __FILE__ ) . '/admin/settings/class-wc-ts-gzd-settings-tab.php'; + $tabs['trusted_shops'] = 'WC_TS_GZD_Settings_Tab'; + return $tabs; + } + + /** + * Add notices + styles if needed. + */ + public function add_notices() { + $screen = function_exists( 'get_current_screen' ) ? get_current_screen() : false; + $screen_id = $screen ? $screen->id : ''; + $show_on_screens = array( + 'dashboard', + 'plugins', + ); + + $wc_screen_ids = function_exists( 'wc_get_screen_ids' ) ? wc_get_screen_ids() : array(); + + // Notices should only show on WooCommerce screens, the main dashboard, and on the plugins screen. + if ( ! in_array( $screen_id, $wc_screen_ids, true ) && ! in_array( $screen_id, $show_on_screens, true ) ) { + return; + } + + if ( 1 === (int) get_option( '_wc_ts_needs_update' ) ) { + if ( current_user_can( 'manage_woocommerce' ) ) { + wp_enqueue_style( 'woocommerce-activation', plugins_url( '/assets/css/activation.css', WC_PLUGIN_FILE ), array(), WC_TRUSTED_SHOPS_VERSION ); + wp_enqueue_style( 'woocommerce-ts-activation', plugins_url( '/assets/css/activation.css', WC_TRUSTED_SHOPS_PLUGIN_FILE ), array(), WC_TRUSTED_SHOPS_VERSION ); + + add_action( 'admin_notices', array( $this, 'install_notice' ) ); + } + } + } + + /** + * Show the install notices + */ + public function install_notice() { + + // If we need to update, include a message with the update button + if ( 1 === (int) get_option( '_wc_ts_needs_update' ) ) { + include WC_TRUSTED_SHOPS_ABSPATH . 'includes/admin/views/html-notice-update.php'; + } + } + + public function toggle_input( $value ) { + // Custom attribute handling. + $custom_attributes = array(); + + if ( ! empty( $value['custom_attributes'] ) && is_array( $value['custom_attributes'] ) ) { + foreach ( $value['custom_attributes'] as $attribute => $attribute_value ) { + $custom_attributes[] = esc_attr( $attribute ) . '="' . esc_attr( $attribute_value ) . '"'; + } + } + + // Description handling. + $field_description = WC_Admin_Settings::get_field_description( $value ); + $description = $field_description['description']; + $tooltip_html = $field_description['tooltip_html']; + $option_value = WC_Admin_Settings::get_option( $value['id'], $value['default'] ); + ?> + + + + + + + + + + /> + + + plugin_path() . '/templates/' . $template ) ) { + $core_file = $this->plugin_path() . '/templates/' . $template; + } + + return $core_file; + } + + public function set_woocommerce_template_dir( $dir, $template ) { + if ( file_exists( WC_trusted_shops()->plugin_path() . '/templates/' . $template ) ) { + return 'woocommerce-trusted-shops'; + } + + return $dir; + } + + public function admin_footer_text( $footer_text ) { + if ( ! current_user_can( 'manage_woocommerce' ) ) { + return; + } + + // Check to make sure we're on a WooCommerce admin page + if ( isset( $_GET['tab'] ) && 'trusted-shops' === $_GET['tab'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $footer_text = wp_kses_post( sprintf( _x( 'If the App helped you, please leave a %1$s★★★★★%2$s in the WordPress plugin repository.', 'trusted-shops', 'woocommerce-germanized' ), '', '' ) ); + } + + return $footer_text; + } + + /** + * Auto-load WC_Trusted_Shops classes on demand to reduce memory consumption. + * + * @param mixed $class + * @return void + */ + public function autoload( $class ) { + $class = strtolower( $class ); + $path = $this->plugin_path() . '/includes/'; + + if ( 0 !== strpos( $class, 'wc_ts_' ) && 0 !== strpos( $class, 'wc_trusted_shops' ) ) { + return; + } + + $file = 'class-' . str_replace( '_', '-', $class ) . '.php'; + + if ( strpos( $class, 'wc_ts_compatibility' ) !== false ) { + $path = $this->plugin_path() . '/includes/compatibility/'; + } + + if ( $path && is_readable( $path . $file ) ) { + include_once $path . $file; + return; + } + } + + /** + * Get the plugin url. + * + * @return string + */ + public function plugin_url() { + return Package::get_url(); + } + + /** + * Get the plugin path. + * + * @return string + */ + public function plugin_path() { + return untrailingslashit( Package::get_path() ); + } + + /** + * Get the language path + * + * @return string + */ + public function language_path() { + return $this->plugin_path() . '/i18n/languages'; + } + + /** + * Define WC_Germanized Constants + */ + private function define_constants() { + define( 'WC_TRUSTED_SHOPS_PLUGIN_FILE', trailingslashit( Package::get_path() ) . 'woocommerce-trusted-shops.php' ); + define( 'WC_TRUSTED_SHOPS_ABSPATH', trailingslashit( Package::get_path() ) ); + define( 'WC_TRUSTED_SHOPS_VERSION', $this->version ); + } + + public function setup_compatibility() { + $plugins = apply_filters( + 'woocommerce_ts_compatibilities', + array( + 'wpml-string-translation', + ) + ); + foreach ( $plugins as $comp ) { + $classname = str_replace( ' ', '_', 'WC_TS_Compatibility_' . ucwords( str_replace( '-', ' ', $comp ) ) ); + if ( class_exists( $classname ) ) { + $this->compatibilities[ $comp ] = new $classname(); + } + } + } + + public function get_compatibility( $name ) { + return ( isset( $this->compatibilities[ $name ] ) ? $this->compatibilities[ $name ] : false ); + } + + /** + * Include required core files used in admin and on the frontend. + */ + private function includes() { + include_once WC_TRUSTED_SHOPS_ABSPATH . 'includes/abstracts/abstract-wc-ts-compatibility.php'; + include_once WC_TRUSTED_SHOPS_ABSPATH . 'includes/class-wc-ts-install.php'; + } + + /** + * Filter WooCommerce Templates to look into /templates before looking within theme folder + * + * @param string $template + * @param string $template_name + * @param string $template_path + * @return string + */ + public function filter_templates( $template, $template_name, $template_path ) { + if ( ! $template_path ) { + $template_path = WC()->template_path(); + } + + // Make filter gzd_compatible + $template_name = apply_filters( 'woocommerce_trusted_shops_template_name', $template_name ); + + // Check Theme + $theme_template = locate_template( + array( + trailingslashit( $template_path ) . $template_name, + $template_name, + ) + ); + + // Load Default + if ( ! $theme_template ) { + if ( file_exists( $this->plugin_path() . '/templates/' . $template_name ) ) { + $template = $this->plugin_path() . '/templates/' . $template_name; + } + } else { + $template = $theme_template; + } + + return apply_filters( 'woocommerce_trusted_shops_filter_template', $template, $template_name, $template_path ); + } + + /** + * Load Localisation files for WooCommerce Germanized. + */ + public function load_plugin_textdomain() { + $locale = is_admin() && function_exists( 'get_user_locale' ) ? get_user_locale() : get_locale(); + $locale = apply_filters( 'plugin_locale', $locale, 'woocommerce-germanized' ); + unload_textdomain( 'woocommerce-trusted-shops' ); + load_textdomain( 'woocommerce-trusted-shops', trailingslashit( WP_LANG_DIR ) . 'woocommerce-trusted-shops/woocommerce-trusted-shops-' . $locale . '.mo' ); + load_plugin_textdomain( 'woocommerce-trusted-shops', false, plugin_basename( dirname( __FILE__ ) ) . '/i18n/languages/' ); + } + + /** + * Show action links on the plugin screen + * + * @param mixed $links + * @return array + */ + public function action_links( $links ) { + return array_merge( + array( + '' . _x( 'Settings', 'trusted-shops', 'woocommerce-germanized' ) . '', + ), + $links + ); + } + + /** + * Add custom styles to Admin + */ + public function add_admin_styles() { + $screen = get_current_screen(); + + if ( isset( $_GET['tab'] ) && 'trusted-shops' === $_GET['tab'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + do_action( 'woocommerce_trusted_shops_load_admin_scripts' ); + } + } + + /** + * Add WooCommerce Germanized Settings Tab + * + * @param array $integrations + * @return array + */ + public function add_settings( $integrations ) { + $integrations[] = new WC_TS_Settings_Handler(); + + return $integrations; + } + + /** + * Add Custom Email templates + * + * @param array $mails + * @return array + */ + public function add_emails( $mails ) { + $mails['WC_TS_Email_Customer_Trusted_Shops'] = include_once $this->plugin_path() . '/includes/emails/class-wc-ts-email-customer-trusted-shops.php'; + + return $mails; + } + + public function add_wpml_emails( $mails ) { + $mails['WC_TS_Email_Customer_Trusted_Shops'] = 'customer_trusted_shops'; + + return $mails; + } + } + +endif; + +/** + * Returns the global instance of WooCommerce Germanized + */ +function WC_trusted_shops() { // phpcs:ignore WordPress.NamingConventions.ValidFunctionName.FunctionNameInvalid + return WooCommerce_Trusted_Shops::instance(); +} + +$GLOBALS['woocommerce_trusted_shops'] = WC_trusted_shops(); diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-review-exporter.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-review-exporter.php new file mode 100644 index 000000000..358958b81 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-review-exporter.php @@ -0,0 +1,171 @@ +statuses = array_keys( wc_get_order_statuses() ); + $this->column_names = $this->get_default_column_names(); + } + + /** + * Return an array of columns to export. + * + * @since 3.1.0 + * @return array + */ + public function get_default_column_names() { + return apply_filters( + "woocommerce_gzd_{$this->export_type}_default_columns", + array( + 'id' => 'reference', + 'date' => 'date', + 'days' => 'days', + 'billing_email' => 'email', + 'billing_first_name' => 'firstName', + 'billing_last_name' => 'lastName', + ) + ); + } + + public function get_interval_days() { + return absint( $this->days_interval ); + } + + public function set_interval_days( $days ) { + $this->days_interval = absint( $days ); + } + + public function get_days_until_send() { + return $this->days_to_send; + } + + public function set_days_until_send( $days ) { + $this->days_to_send = absint( $days ); + } + + public function get_statuses() { + return $this->statuses; + } + + public function set_statuses( $statuses ) { + $this->statuses = (array) $statuses; + } + + public function set_lang( $lang ) { + $this->lang = $lang; + } + + public function get_lang() { + return $this->lang; + } + + /** + * Prepare data that will be exported. + */ + public function prepare_data_to_export() { + $columns = $this->get_column_names(); + $date = date( 'Y-m-d', strtotime( '-' . $this->get_interval_days() . ' days' ) ); // phpcs:ignore WordPress.DateTime.RestrictedFunctions.date_date + $args = array( + 'post_type' => 'shop_order', + 'post_status' => $this->get_statuses(), + 'showposts' => -1, + 'date_query' => array( + array( + 'after' => $date, + ), + ), + ); + + if ( $this->get_lang() !== '' && 'all' !== $this->get_lang() ) { + $args['meta_query'] = array(); // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query + $args['meta_query']['wpml'] = array( + 'key' => 'wpml_language', + 'compare' => '=', + 'value' => $this->get_lang(), + ); + } + + $order_query = new WP_Query( apply_filters( "woocommerce_gzd_{$this->export_type}_query_args", $args ) ); + $this->total_rows = $order_query->found_posts; + $this->row_data = array(); + + while ( $order_query->have_posts() ) { + $order_query->next_post(); + + $order = wc_get_order( $order_query->post->ID ); + $row = array(); + + foreach ( $columns as $column_id => $column_name ) { + $column_id = strstr( $column_id, ':' ) ? current( explode( ':', $column_id ) ) : $column_id; + $value = ''; + + if ( is_callable( array( $this, "get_column_value_{$column_id}" ) ) ) { + // Handle special columns which don't map 1:1 to order data. + $value = $this->{"get_column_value_{$column_id}"}( $order ); + + } elseif ( wc_ts_get_crud_data( $order, $column_id ) ) { + // Default and custom handling. + $value = wc_ts_get_crud_data( $order, $column_id ); + } + + $row[ $column_id ] = $value; + } + + $this->row_data[] = apply_filters( 'woocommerce_gzd_trusted_shops_review_export_row_data', $row, $order ); + } + } + + public function get_column_value_date( $order ) { + return wc_ts_get_order_date( $order, 'd.m.Y' ); + } + + public function get_column_value_days( $order ) { + return $this->get_days_until_send(); + } +} diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-schedule.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-schedule.php new file mode 100644 index 000000000..804e764c0 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-schedule.php @@ -0,0 +1,246 @@ +base = $base; + + add_action( 'woocommerce_gzd_trusted_shops_reviews', array( $this, 'update_reviews' ) ); + add_action( 'admin_init', array( $this, 'update_default_reviews' ), 10 ); + add_action( 'woocommerce_gzd_trusted_shops_reviews', array( $this, 'send_mails' ) ); + } + + public function update_default_reviews() { + // Generate reviews for the first time + $option_key = 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_reviews_cache'; + + $section = isset( $_GET['section'] ) ? wc_clean( wp_unslash( $_GET['section'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + // Do only update default reviews if the admin user open the settings page. + if ( 'trusted_shops' !== $section ) { + return; + } + + if ( ! get_option( $option_key, false ) ) { + $this->update_reviews_inner(); + } + + if ( $this->base->is_multi_language_setup() ) { + $compatibility = $this->base->get_multi_language_compatibility(); + $current_language = $compatibility->get_current_language(); + + global $wc_ts_original_lang; + + $wc_ts_original_lang = $current_language; + + foreach ( $compatibility->get_languages() as $language ) { + + if ( $compatibility->get_default_language() === $language ) { + continue; + } + + $option_key .= '_' . $language; + + if ( ! get_option( $option_key, false ) ) { + $this->update_reviews_inner( $language ); + } + } + } + } + + /** + * Update Review Cache by grabbing information from xml file + */ + public function update_reviews() { + $this->update_reviews_inner(); + + if ( $this->base->is_multi_language_setup() ) { + + $compatibility = $this->base->get_multi_language_compatibility(); + $current_language = $compatibility->get_current_language(); + + global $wc_ts_original_lang; + + $wc_ts_original_lang = $current_language; + + foreach ( $compatibility->get_languages() as $language ) { + if ( $compatibility->get_default_language() === $language ) { + continue; + } + + $this->update_reviews_inner( $language ); + } + } + } + + protected function update_reviews_inner( $lang = '' ) { + if ( ! empty( $lang ) ) { + wc_ts_switch_language( $lang ); + } + + if ( ! $this->base->is_rich_snippets_enabled() ) { + if ( ! empty( $lang ) ) { + wc_ts_restore_language(); + } + + return; + } + + $update = array(); + + if ( $this->base->is_enabled() ) { + + $response = wp_remote_post( $this->base->api_url ); + + if ( is_array( $response ) ) { + $output = json_decode( $response['body'], true ); + + if ( isset( $output['response']['data'] ) ) { + $reviews = $output['response']['data']['shop']['qualityIndicators']['reviewIndicator']; + $update['count'] = (string) $reviews['activeReviewCount']; + $update['avg'] = (float) $reviews['overallMark']; + $update['max'] = '5.00'; + } + } + } + + $option_key = 'woocommerce_' . $this->base->option_prefix . 'trusted_shops_reviews_cache'; + + if ( ! empty( $lang ) ) { + $option_key .= '_' . $lang; + } + + update_option( $option_key, $update ); + + if ( ! empty( $lang ) ) { + wc_ts_restore_language(); + } + } + + /** + * Placeholder to avoid fatal errors within scheduled actions. + * + * @deprecated 2.2.5 + */ + public function update_review_widget() {} + + /** + * Send review reminder mails after x days + */ + public function send_mails() { + if ( $this->base->is_multi_language_setup() ) { + $compatibility = $this->base->get_multi_language_compatibility(); + $current_language = $compatibility->get_current_language(); + + global $wc_ts_original_lang; + + $wc_ts_original_lang = $current_language; + + foreach ( $compatibility->get_languages() as $language ) { + $this->send_mails_inner( $language ); + } + } else { + $this->send_mails_inner(); + } + } + + protected function send_mails_inner( $lang = '' ) { + if ( ! empty( $lang ) ) { + wc_ts_switch_language( $lang ); + } + + if ( ! $this->base->is_review_reminder_enabled() ) { + if ( ! empty( $lang ) ) { + wc_ts_restore_language(); + } + + return; + } + + $order_statuses = $this->base->review_reminder_status; + + if ( ! is_array( $order_statuses ) ) { + $order_statuses = array( $order_statuses ); + } + + $args = array( + 'post_type' => 'shop_order', + 'post_status' => apply_filters( 'woocommerce_trusted_shops_review_reminder_valid_order_statuses', $order_statuses ), + 'showposts' => -1, + 'meta_query' => array( // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query + 'relation' => 'AND', + 'is_sent' => array( + 'key' => '_trusted_shops_review_mail_sent', + 'compare' => 'NOT EXISTS', + ), + 'opted_in' => array( + 'key' => '_ts_review_reminder_opted_in', + 'compare' => '=', + 'value' => 'yes', + ), + ), + ); + + if ( ! empty( $lang ) ) { + $args['meta_query']['wpml'] = array( + 'key' => 'wpml_language', + 'compare' => '=', + 'value' => $lang, + ); + } + + $order_query = new WP_Query( apply_filters( 'woocommerce_trusted_shops_review_reminder_order_args', $args, $lang ) ); + + while ( $order_query->have_posts() ) { + + $order_query->next_post(); + + if ( ! $order = wc_get_order( $order_query->post->ID ) ) { + continue; + } + + $completed_date = apply_filters( 'woocommerce_trusted_shops_review_reminder_order_completed_date', $order->get_date_completed(), $order ); + + if ( ! $completed_date ) { + continue; + } + + $now = new DateTime(); + $diff = $now->diff( $completed_date ); + $min_days = (int) $this->base->review_reminder_days; + + if ( $diff->days >= $min_days ) { + + if ( apply_filters( 'woocommerce_trusted_shops_send_review_reminder_email', true, $order ) ) { + + $mails = WC()->mailer()->get_emails(); + + foreach ( $mails as $mail ) { + + if ( 'customer_trusted_shops' === $mail->id ) { + $mail->trigger( wc_ts_get_crud_data( $order, 'id' ) ); + + update_post_meta( wc_ts_get_crud_data( $order, 'id' ), '_trusted_shops_review_mail_sent', 1 ); + } + } + } + } + } + + if ( ! empty( $lang ) ) { + wc_ts_restore_language(); + } + } +} diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-shortcodes.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-shortcodes.php new file mode 100644 index 000000000..0299f9331 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-shortcodes.php @@ -0,0 +1,99 @@ +base = $base; + + add_action( 'init', array( $this, 'init' ), 3 ); + } + + public function init() { + + // Define shortcodes + $shortcodes = array( + 'trusted_shops_rich_snippets' => array( $this, 'trusted_shops_rich_snippets' ), + 'trusted_shops_review_sticker' => array( $this, 'trusted_shops_review_sticker' ), + 'trusted_shops_badge' => array( $this, 'trusted_shops_badge' ), + ); + + foreach ( $shortcodes as $shortcode => $function ) { + add_shortcode( apply_filters( "{$shortcode}_shortcode_tag", $shortcode ), $function ); + } + + } + + /** + * Returns Trusted Shops rich snippet review html + * + * @param array $atts + * @return string + */ + public function trusted_shops_rich_snippets( $atts ) { + ob_start(); + wc_get_template( + 'trusted-shops/rich-snippets.php', + array( + 'plugin' => $this->base, + ) + ); + $html = ob_get_clean(); + + return $this->base->is_enabled() ? $html : ''; + } + + /** + * Returns Trusted Shops reviews graphic + * + * @param array $atts + * @return string + */ + public function trusted_shops_review_sticker( $atts ) { + $atts = wp_parse_args( + $atts, + array( + 'element' => '#ts_review_sticker', + ) + ); + + ob_start(); + wc_get_template( + 'trusted-shops/review-sticker.php', + array( + 'plugin' => $this->base, + 'element' => $atts['element'], + ) + ); + $html = ob_get_clean(); + return $this->base->is_enabled() ? '
' . $html . '
' : ''; + } + + /** + * Returns Trusted Shops Badge html + * + * @param array $atts + * @return string + */ + public function trusted_shops_badge( $atts ) { + $atts = wp_parse_args( + $atts, + array( + 'width' => 0, + ) + ); + + return $this->base->is_enabled() ? '' : ''; + } + +} diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-template-hooks.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-template-hooks.php new file mode 100644 index 000000000..3932ae4e2 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-template-hooks.php @@ -0,0 +1,280 @@ +base = $base; + + // Load hooks on init so that language-specific settings are loaded + add_action( 'init', array( $this, 'init' ), 10 ); + + // Always register checkbox to avoid language problems + add_action( 'woocommerce_gzd_register_legal_core_checkboxes', array( $this, 'review_reminder_checkbox' ), 30 ); + } + + public function init() { + if ( $this->base->is_enabled() ) { + add_action( 'woocommerce_thankyou', array( $this, 'template_thankyou' ), 10, 1 ); + add_action( 'wp_footer', array( $this, 'template_trustbadge' ), 250 ); + } + + if ( $this->base->product_reviews_visible() ) { + add_filter( 'woocommerce_product_tabs', array( $this, 'remove_review_tab' ), 40, 1 ); + } + + if ( $this->base->is_product_sticker_enabled() ) { + add_filter( 'woocommerce_product_tabs', array( $this, 'review_tab' ), 50, 1 ); + } + + if ( $this->base->is_product_widget_enabled() ) { + add_filter( 'woocommerce_trusted_shops_template_name', array( $this, 'set_product_widget_template' ), 50, 1 ); + } + + if ( $this->base->is_rich_snippets_enabled() ) { + add_action( 'wp_footer', array( $this, 'insert_rich_snippets' ), 20 ); + } + + // Save Fields on order + if ( $this->base->is_review_reminder_enabled() ) { + add_action( 'woocommerce_checkout_update_order_meta', array( $this, 'update_order_meta' ) ); + + if ( 'yes' === $this->base->review_reminder_opt_out ) { + // Email notices right beneath order table + add_action( 'woocommerce_email_after_order_table', array( $this, 'email_cancel_review_reminder' ), 8, 3 ); + add_filter( 'woocommerce_email_styles', array( $this, 'email_styles' ) ); + + // Check for customer activation + add_action( 'template_redirect', array( $this, 'cancel_review_reminder_check' ) ); + } + } + } + + public function insert_rich_snippets() { + $insert = false; + + if ( in_array( 'category', $this->base->get_rich_snippets_locations(), true ) ) { + if ( is_product_category() ) { + $insert = true; + } + } + + if ( in_array( 'home', $this->base->get_rich_snippets_locations(), true ) ) { + if ( function_exists( 'is_shop' ) && is_shop() ) { + $insert = true; + } + } + + if ( in_array( 'product', $this->base->get_rich_snippets_locations(), true ) ) { + if ( is_product() ) { + $insert = true; + } + } + + if ( $insert ) { + echo do_shortcode( '[trusted_shops_rich_snippets]' ); + } + } + + public function cancel_review_reminder_check() { + if ( isset( $_GET['disable-review-reminder'] ) && isset( $_GET['order-id'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + $order_id = absint( $_GET['order-id'] ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $code = wc_clean( wp_unslash( $_GET['disable-review-reminder'] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended + + if ( ! empty( $code ) && ! empty( $order_id ) ) { + + $order_query = new WP_Query( + array( + 'post_type' => 'shop_order', + 'p' => $order_id, + 'post_status' => array_keys( wc_get_order_statuses() ), + 'posts_per_page' => 1, + 'meta_query' => array( // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query + 'code' => array( + 'key' => '_ts_cancel_review_reminder_code', + 'compare' => '=', + 'value' => $code, + ), + ), + ) + ); + + while ( $order_query->have_posts() ) { + $order_query->next_post(); + $order = wc_get_order( $order_query->post->ID ); + + if ( $order ) { + $order_id = wc_ts_get_crud_data( $order, 'id' ); + + delete_post_meta( $order_id, '_ts_cancel_review_reminder_code' ); + delete_post_meta( $order_id, '_ts_review_reminder_opted_in' ); + + wp_die( wp_kses_post( sprintf( _x( 'Your review reminder e-mail has been cancelled successfully. Return to %s.', 'trusted-shops', 'woocommerce-germanized' ), '' . esc_html_x( 'Home', 'trusted-shops', 'woocommerce-germanized' ) . '' ) ) ); + } + } + } + } + } + + public function email_styles( $css ) { + $css .= ' + .wc-ts-cancel-review-reminder { + margin-top: 16px; + } + '; + + return $css; + } + + public function get_cancel_review_reminder_link( $order ) { + $code = wc_ts_get_crud_data( $order, 'ts_cancel_review_reminder_code' ); + + if ( ! $code || empty( $code ) ) { + global $wp_hasher; + + if ( empty( $wp_hasher ) ) { + require_once ABSPATH . WPINC . '/class-phpass.php'; + $wp_hasher = new PasswordHash( 8, true ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited + } + + $code = $wp_hasher->HashPassword( wp_generate_password( 20 ) ); + + update_post_meta( wc_ts_get_crud_data( $order, 'id' ), '_ts_cancel_review_reminder_code', $code ); + } + + $order_id = wc_ts_get_crud_data( $order, 'id' ); + $link = add_query_arg( + array( + 'disable-review-reminder' => $code, + 'order-id' => $order_id, + ), + get_site_url() + ); + + if ( $lang = wc_ts_get_order_language( $order ) ) { + $link = add_query_arg( array( 'lang' => $lang ), $link ); + } + + return esc_url_raw( apply_filters( 'woocommerce_trusted_shops_cancel_review_reminder_link', $link, $code, $order ) ); + } + + public function email_cancel_review_reminder( $order, $sent_to_admin, $plain_text ) { + $type = WC_germanized()->emails->get_current_email_object(); + + // Try to flush the cache before continuing + WC_GZD_Cache_Helper::maybe_flush_cache( + 'db', + array( + 'cache_type' => 'meta', + 'meta_type' => 'post', + 'meta_key' => 'ts_review_reminder_opted_in', // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key + ) + ); + $opted_in = wc_ts_get_crud_data( $order, 'ts_review_reminder_opted_in' ); + + if ( $type && 'yes' === $opted_in && 'customer_processing_order' === $type->id ) { + wc_get_template( 'emails/cancel-review-reminder.php', array( 'link' => $this->get_cancel_review_reminder_link( $order ) ) ); + } + } + + public function update_order_meta( $order_id ) { + $checkbox = wc_gzd_get_legal_checkbox( 'review_reminder' ); + + if ( isset( $_POST['review_reminder'] ) || ! $checkbox || ( $checkbox && ! $checkbox->is_enabled() ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing + update_post_meta( $order_id, '_ts_review_reminder_opted_in', 'yes' ); + } + } + + public function review_reminder_checkbox() { + if ( ! function_exists( 'wc_gzd_register_legal_checkbox' ) ) { + return; + } + + wc_gzd_register_legal_checkbox( + 'review_reminder', + array( + 'html_id' => 'review-reminder', + 'html_name' => 'review_reminder', + 'html_wrapper_classes' => array( 'legal' ), + 'label' => _x( 'Yes, I would like to be reminded via e-mail after {days} day(s) to review my order. I am able to cancel the reminder at any time by clicking on the "cancel review reminder" link within the order confirmation.', 'trusted-shops', 'woocommerce-germanized' ), + 'label_args' => array( '{days}' => $this->base->review_reminder_days ), + 'hide_input' => false, + 'is_enabled' => false, + 'is_mandatory' => false, + 'error_message' => _x( 'Please allow us to send a review reminder by e-mail.', 'trusted-shops', 'woocommerce-germanized' ), + 'priority' => 6, + 'is_core' => true, + 'admin_name' => _x( 'Review reminder', 'trusted-shops', 'woocommerce-germanized' ), + 'admin_desc' => _x( 'Asks the customer to receive a Trusted Shops review reminder.', 'trusted-shops', 'woocommerce-germanized' ), + 'locations' => array( 'checkout' ), + ) + ); + } + + public function set_product_widget_template( $template ) { + + if ( in_array( $template, array( 'single-product/rating.php' ), true ) ) { + $template = 'trusted-shops/product-widget.php'; + } + + return $template; + } + + public function remove_review_tab( $tabs ) { + if ( isset( $tabs['reviews'] ) ) { + unset( $tabs['reviews'] ); + } + + return $tabs; + } + + public function review_tab( $tabs ) { + $tabs['trusted_shops_reviews'] = array( + 'title' => $this->base->product_sticker_tab_text, + 'priority' => 30, + 'callback' => array( $this, 'template_product_sticker' ), + ); + return $tabs; + } + + public function template_review_sticker( $template ) { + wc_get_template( + 'trusted-shops/review-sticker.php', + array( + 'plugin' => $this->base, + 'element' => '#ts_review_sticker', + ) + ); + } + + public function template_product_sticker( $template ) { + wc_get_template( 'trusted-shops/product-sticker.php', array( 'plugin' => $this->base ) ); + } + + public function template_trustbadge() { + wc_get_template( 'trusted-shops/trustbadge.php', array( 'plugin' => $this->base ) ); + } + + public function template_thankyou( $order_id ) { + wc_get_template( + 'trusted-shops/thankyou.php', + array( + 'order_id' => $order_id, + 'plugin' => $this->base, + ) + ); + } + +} diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-widgets.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-widgets.php new file mode 100644 index 000000000..107121ed3 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops-widgets.php @@ -0,0 +1,34 @@ +base = $base; + + add_action( 'widgets_init', array( $this, 'include_widgets' ), 25 ); + } + + public function include_widgets() { + if ( $this->base->is_review_sticker_enabled() ) { + $this->register_widget( 'review_sticker' ); + } + } + + private function register_widget( $name ) { + $classname = $this->base->get_dependency_name( 'widget_' . $name ); + include_once 'widgets/class-' . strtolower( str_replace( '_', '-', $classname ) ) . '.php'; + register_widget( $classname ); + } + +} diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops.php b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops.php new file mode 100644 index 000000000..9be75601a --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-trusted-shops.php @@ -0,0 +1,689 @@ +plugin = $plugin; + + $args = wp_parse_args( + $params, + array( + 'et_params' => array(), + 'signup_params' => array(), + 'prefix' => '', + 'signup_url' => '', + 'supports' => array( 'reminder' ), + 'path' => dirname( __FILE__ ) . '/', + ) + ); + + foreach ( $args as $arg => $val ) { + $this->$arg = $val; + } + + $this->option_prefix = strtolower( $this->prefix ); + $this->load(); + } + + public function load() { + + // Refresh TS ID + API URL + $this->refresh(); + $this->includes(); + + add_action( 'init', array( $this, 'refresh' ), 50 ); + + if ( is_admin() ) { + $this->get_dependency( 'admin' ); + } + + $this->get_dependency( 'schedule' ); + $this->get_dependency( 'shortcodes' ); + $this->get_dependency( 'widgets' ); + $this->get_dependency( 'template_hooks' ); + + if ( $this->is_enabled() ) { + add_action( 'wp_enqueue_scripts', array( $this, 'load_frontend_assets' ), 50 ); + + if ( is_admin() ) { + add_filter( 'woocommerce_gzd_wpml_translatable_options', array( $this, 'register_wpml_options' ), 20, 1 ); + add_filter( 'woocommerce_gzd_wpml_remove_translation_empty_equal', array( $this, 'stop_wpml_options_string_deletions' ), 20, 4 ); + } + } + } + + public function register_wpml_options( $settings ) { + $admin = $this->get_dependency( 'admin' ); + + return array_merge( $settings, $admin->get_translatable_settings() ); + } + + /** + * Make sure that other languages are not synced with main language e.g. option does not default to main language + * + * @param $allow + * @param $option + * @param $new_value + * @param $old_value + * @return bool + */ + public function stop_wpml_options_string_deletions( $allow, $option, $new_value, $old_value ) { + $admin = $this->get_dependency( 'admin' ); + + if ( array_key_exists( $option, $admin->get_translatable_settings() ) ) { + $allow = false; + } + + return $allow; + } + + public function includes() { + include_once $this->path . 'wc-ts-core-functions.php'; + } + + public function load_frontend_assets() { + $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; + $assets_path = $this->plugin->plugin_url() . '/assets/css'; + + wp_register_style( 'woocommerce-trusted-shops', $assets_path . '/layout' . $suffix . '.css', false, $this->plugin->version ); + wp_enqueue_style( 'woocommerce-trusted-shops' ); + } + + public function get_dependency_name( $name ) { + $classname = 'WC_Trusted_Shops_' . ucwords( str_replace( '-', '_', strtolower( $name ) ) ); + return $classname; + } + + public function get_dependency( $name ) { + $classname = $this->get_dependency_name( $name ); + + return call_user_func_array( array( $classname, 'instance' ), array( $this ) ); + } + + public function refresh() { + $this->id = $this->__get( 'id' ); + $this->api_url = 'http://api.trustedshops.com/rest/public/v2/shops/' . $this->id . '/quality.json'; + } + + public function get_multi_language_compatibility() { + return apply_filters( 'woocommerce_trusted_shops_multi_language_compatibility', $this->plugin->get_compatibility( 'wpml-string-translation' ) ); + } + + public function is_multi_language_setup() { + $compatibility = $this->get_multi_language_compatibility(); + + return $compatibility->is_activated() ? true : false; + } + + /** + * Get Trusted Shops Options + * + * @param string $key + * @return mixed + */ + public function __get( $key ) { + $option_name = 'woocommerce_' . $this->option_prefix . 'trusted_shops_' . $key; + $value = get_option( $option_name ); + + /** + * By default WPML does not allow empty strings to override default translations. + * This snippet manually checks for translations and allows to override default WPML translations. + */ + if ( ! is_admin() && $this->is_multi_language_setup() ) { + $compatibility = $this->get_multi_language_compatibility(); + + $default_language = $compatibility->get_default_language(); + $current_language = $compatibility->get_current_language(); + + if ( $current_language !== $default_language ) { + if ( isset( $this->options[ $current_language ][ $key ] ) ) { + return $this->options[ $current_language ][ $key ]; + } else { + if ( $string_id = $compatibility->get_string_id( $option_name ) ) { + $translation = $compatibility->get_string_translation( $string_id, $current_language ); + + if ( false !== $translation ) { + $this->options[ $current_language ][ $key ] = $translation; + $value = $translation; + } + } + } + } + } + + return $value; + } + + /** + * Checks whether a certain Trusted Shops Option isset + * + * @param string $key + * @return boolean + */ + public function __isset( $key ) { + return ( ! get_option( 'woocommerce_' . $this->option_prefix . 'trusted_shops_' . $key ) ) ? false : true; + } + + /** + * Checks whether Trusted Shops is enabled + * + * @return boolean + */ + public function is_enabled() { + return ( $this->id ) ? true : false; + } + + public function get_rich_snippets_locations() { + $locations = array(); + + if ( 'yes' === $this->rich_snippets_category ) { + $locations[] = 'category'; + } + + if ( 'yes' === $this->rich_snippets_product ) { + $locations[] = 'product'; + } + + if ( 'yes' === $this->rich_snippets_home ) { + $locations[] = 'home'; + } + + return $locations; + } + + public function is_trustbadge_enabled() { + return ( 'yes' === $this->trustbadge_enable && '' !== $this->id ? true : false ); + } + + /** + * Checks whether Trusted Shops Rich Snippets are enabled + * + * @return boolean + */ + public function is_rich_snippets_enabled() { + return ( 'yes' === $this->rich_snippets_enable && $this->is_enabled() ? true : false ); + } + + /** + * Checks whether review widget is enabled + * + * @return boolean + */ + public function is_review_widget_enabled() { + return ( 'yes' === $this->review_widget_enable && $this->is_enabled() ? true : false ); + } + + public function is_review_reminder_enabled() { + return ( 'yes' === $this->review_reminder_enable && $this->supports( 'reminder' ) && $this->is_enabled() ? true : false ); + } + + public function is_review_reminder_checkbox_enabled() { + return ( 'yes' === $this->review_reminder_checkbox && $this->is_review_reminder_enabled() ? true : false ); + } + + public function product_reviews_visible() { + return ( $this->is_enabled() && $this->is_product_sticker_enabled() ? true : false ); + } + + public function is_product_reviews_enabled() { + return ( 'yes' === $this->reviews_enable && $this->is_enabled() ? true : false ); + } + + public function is_product_sticker_enabled() { + return ( $this->is_product_reviews_enabled() && 'yes' === $this->product_sticker_enable ? true : false ); + } + + public function is_review_sticker_enabled() { + return 'yes' === $this->review_sticker_enable ? true : false; + } + + public function is_product_widget_enabled() { + return ( $this->is_product_reviews_enabled() && 'yes' === $this->product_widget_enable ? true : false ); + } + + public function supports( $type ) { + return ( in_array( $type, $this->supports, true ) ? true : false ); + } + + /** + * Gets Trusted Shops payment gateway by woocommerce payment id + * + * @param integer $payment_method_id + * @return string + */ + public function get_payment_gateway( $payment_method_id ) { + $gateways = WC()->payment_gateways()->payment_gateways(); + + if ( array_key_exists( $payment_method_id, $gateways ) ) { + return $gateways[ $payment_method_id ]->get_method_title(); + } else { + return 'wcOther'; + } + } + + /** + * Returns the average rating by grabbing the rating from the current languages' cache. + * + * @return array + */ + public function get_average_rating() { + $reviews = ( $this->reviews_cache ? $this->reviews_cache : array() ); + + if ( $this->is_multi_language_setup() ) { + $default_language = $this->get_multi_language_compatibility()->get_default_language(); + $current_language = $this->get_multi_language_compatibility()->get_current_language(); + + if ( $current_language !== $default_language ) { + $reviews = ( $this->{"reviews_cache_{$current_language}"} ? $this->{"reviews_cache_{$current_language}"} : array() ); + } + } + + return $reviews; + } + + /** + * Returns the certificate link + * + * @return string + */ + public function get_certificate_link() { + return 'https://www.trustedshops.com/shop/certificate.php?shop_id=' . $this->id; + } + + /** + * Returns add new rating link + * + * @return string + */ + public function get_new_review_link( $email, $order_id ) { + return 'https://www.trustedshops.de/bewertung/bewerten_' . $this->id . '.html&buyerEmail=' . rawurlencode( base64_encode( $email ) ) . '&shopOrderID=' . rawurlencode( base64_encode( $order_id ) ); // phpcs:ignore WordPress.PHP.DiscouragedPHPFunctions.obfuscation_base64_encode + } + + /** + * Returns the rating link + * + * @return string + */ + public function get_rating_link() { + return 'https://www.trustedshops.de/bewertung/info_' . $this->id . '.html'; + } + + /** + * Gets the attachment id of review widget graphic + * + * @return mixed + */ + public function get_review_widget_attachment() { + return ( ! $this->review_widget_attachment ? false : $this->review_widget_attachment ); + } + + protected function get_product_shopping_data( $id, $attribute ) { + $product = is_numeric( $id ) ? wc_get_product( $id ) : $id; + + if ( ! $product ) { + return false; + } + + $data = wc_ts_get_crud_data( $product, $attribute ); + + if ( empty( $data ) && 'variation' === $product->get_type() ) { + if ( $parent = wc_get_product( wc_ts_get_crud_data( $product, 'parent' ) ) ) { + $data = wc_ts_get_crud_data( $parent, $attribute ); + } + } + + return $data; + } + + public function get_product_image( $id ) { + $product = is_numeric( $id ) ? wc_get_product( $id ) : $id; + + if ( ! $product ) { + return false; + } + + $image = ''; + + if ( is_callable( array( $product, 'get_image_id' ) ) ) { + $image_id = $product->get_image_id(); + $images = wp_get_attachment_image_src( $image_id, 'shop_single' ); + + if ( ! empty( $images ) ) { + $image = $images[0]; + } + } else { + if ( has_post_thumbnail( wc_ts_get_crud_data( $product, 'id' ) ) ) { + $images = wp_get_attachment_image_src( get_post_thumbnail_id( wc_ts_get_crud_data( $product, 'id' ) ), 'shop_single' ); + + if ( ! empty( $images ) ) { + $image = $images[0]; + } + } + } + + return $image; + } + + public function get_product_brand( $id ) { + $product = is_numeric( $id ) ? wc_get_product( $id ) : $id; + + if ( ! $product ) { + return false; + } + + return $product->get_attribute( $this->brand_attribute ); + } + + public function get_product_mpn( $id ) { + return $this->get_product_shopping_data( $id, '_ts_mpn' ); + } + + public function get_product_gtin( $id ) { + return $this->get_product_shopping_data( $id, '_ts_gtin' ); + } + + public function get_product_skus( $id ) { + $product = is_numeric( $id ) ? wc_get_product( $id ) : $id; + $skus = array(); + $skus[] = ( $product->get_sku() ) ? $product->get_sku() : wc_ts_get_crud_data( $product, 'id' ); + + if ( 'grouped' === $product->get_type() ) { + foreach ( $product->get_children() as $child ) { + if ( $child_product = wc_get_product( $child ) ) { + $skus[] = ( $child_product->get_sku() ) ? $child_product->get_sku() : wc_ts_get_crud_data( $child_product, 'id' ); + } + } + } + + return $skus; + } + + public function get_template( $name ) { + $html = ''; + + ob_start(); + wc_get_template( 'trusted-shops/' . str_replace( '_', '-', $name ) . '-tpl.php', array( 'plugin' => $this ) ); + $html = ob_get_clean(); + + /** + * Use strip_tags instead of wp_strip_all_tags because we want to keep JS. + */ + return preg_replace( '/^\h*\v+/m', '', strip_tags( $html ) ); // phpcs:ignore WordPress.WP.AlternativeFunctions.strip_tags_strip_tags + } + + public function get_script( $name, $replace = true, $args = array() ) { + $script = $this->get_template( $name ); + + if ( 'expert' === $this->integration_mode ) { + $option_script = $this->{$name . '_code'}; + + if ( $option_script ) { + $script = $option_script; + } + } + + if ( $replace ) { + $args = wp_parse_args( + $args, + array( + 'id' => $this->id, + 'locale' => $this->get_locale(), + ) + ); + + foreach ( $args as $key => $arg ) { + $search = '{' . $key . '}'; + $replace = $arg; + + if ( is_array( $arg ) ) { + $search = "'{" . $key . "}'"; + + foreach ( $arg as $k => $v ) { + $arg[ $k ] = "'$v'"; + } + + $replace = implode( ',', $arg ); + } + + $script = str_replace( $search, $replace, $script ); + } + } + + return $script; + } + + public function get_selector_attribute( $type, $selector = '' ) { + $element = $this->get_selector_raw( $type, $selector ); + + if ( substr( $element, 0, 1 ) === '.' ) { + $element = substr( $element, 1 ); + } elseif ( substr( $element, 0, 1 ) === '#' ) { + $element = substr( $element, 1 ); + } + + return $element; + } + + public function get_selector_raw( $type, $selector = '' ) { + $element = $this->{$type . '_selector'}; + + if ( empty( $element ) ) { + $element = "#ts_{$type}"; + } + + if ( ! empty( $selector ) ) { + $element = $selector; + } + + return $element; + } + + public function get_selector( $type, $selector = '' ) { + $element = $this->get_selector_raw( $type, $selector ); + $attribute = $this->get_selector_attribute( $type, $selector ); + $is_class = false; + + if ( substr( $element, 0, 1 ) === '.' ) { + $is_class = true; + } + + return $is_class ? 'class="' . esc_attr( $attribute ) . '"' : 'id="' . esc_attr( $attribute ) . '"'; + } + + public function get_product_sticker_code( $replace = true, $args = array() ) { + if ( $replace ) { + $selector = $this->product_sticker_selector; + + $args = wp_parse_args( + $args, + array( + 'element' => empty( $selector ) ? '#ts_product_sticker' : $selector, + 'border_color' => $this->product_sticker_border_color, + 'star_color' => $this->product_sticker_star_color, + 'star_size' => $this->product_sticker_star_size, + ) + ); + } + + return $this->get_script( 'product_sticker', $replace, $args ); + } + + public function get_review_sticker_code( $replace = true, $args = array() ) { + if ( $replace ) { + $args = wp_parse_args( + $args, + array( + 'element' => '#ts_review_sticker', + 'bg_color' => $this->review_sticker_bg_color, + 'font' => $this->review_sticker_font, + 'number' => $this->review_sticker_number, + 'better_than' => $this->review_sticker_better_than, + ) + ); + } + + return $this->get_script( 'review_sticker', $replace, $args ); + } + + public function get_product_widget_code( $replace = true, $args = array() ) { + if ( $replace ) { + $selector = $this->product_widget_selector; + + $args = wp_parse_args( + $args, + array( + 'element' => empty( $selector ) ? '#ts_product_widget' : $selector, + 'star_color' => $this->product_widget_star_color, + 'star_size' => $this->product_widget_star_size, + 'font_size' => $this->product_widget_font_size, + ) + ); + } + + return $this->get_script( 'product_widget', $replace, $args ); + } + + public function get_trustbadge_code( $replace = true, $args = array() ) { + if ( $replace ) { + $args = wp_parse_args( + $args, + array( + 'offset' => $this->trustbadge_y, + 'variant' => 'standard' === $this->trustbadge_variant ? 'reviews' : 'default', + 'disable' => $this->is_trustbadge_enabled() ? 'false' : 'true', + ) + ); + } + + return $this->get_script( 'trustbadge', $replace, $args ); + } + + public function get_rich_snippets_code( $replace = true, $args = array() ) { + if ( $replace ) { + $rating = $this->get_average_rating(); + + $args = apply_filters( + 'woocommerce_trusted_shops_rich_snippets_args', + wp_parse_args( + $args, + array( + 'average' => $rating['avg'], + 'count' => $rating['count'], + 'maximum' => $rating['max'], + 'rating' => $rating, + 'name' => get_bloginfo( 'name' ), + ) + ), + $this + ); + } + + return $this->get_script( 'rich_snippets', $replace, $args ); + } + + public function get_supported_languages() { + return array_keys( $this->get_locale_mapping() ); + } + + protected function get_locale_mapping() { + $supported = array( + 'de' => 'de_DE', + 'en' => 'en_GB', + 'fr' => 'fr_FR', + ); + + return $supported; + } + + public function get_language() { + $locale = $this->get_locale(); + + return substr( $locale, 0, 2 ); + } + + public function get_locale() { + $supported = $this->get_locale_mapping(); + + $locale = 'en_GB'; + $base = substr( function_exists( 'get_user_locale' ) ? get_user_locale() : get_locale(), 0, 2 ); + + if ( isset( $supported[ $base ] ) ) { + $locale = $supported[ $base ]; + } + + return $locale; + } +} + + diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-ts-dependencies.php b/packages/woocommerce-trusted-shops/includes/class-wc-ts-dependencies.php new file mode 100644 index 000000000..a7ac49fb4 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-ts-dependencies.php @@ -0,0 +1,144 @@ + array( + 'version' => '2.4', + 'version_prefix' => 'woocommerce', + 'name' => 'WooCommerce', + ), + ); + + public static function instance() { + if ( is_null( self::$_instance ) ) { + self::$_instance = new self(); + } + return self::$_instance; + } + + /** + * Cloning is forbidden. + * + * @since 1.0 + */ + public function __clone() { + _doing_it_wrong( __FUNCTION__, esc_html_x( 'Cheating huh?', 'trusted-shops', 'woocommerce-germanized' ), '1.0' ); + } + + /** + * Unserializing instances of this class is forbidden. + * + * @since 1.0 + */ + public function __wakeup() { + _doing_it_wrong( __FUNCTION__, esc_html_x( 'Cheating huh?', 'trusted-shops', 'woocommerce-germanized' ), '1.0' ); + } + + public function __construct() { + + $this->plugins = (array) get_option( 'active_plugins', array() ); + + if ( is_multisite() ) { + $this->plugins = array_merge( $this->plugins, get_site_option( 'active_sitewide_plugins', array() ) ); + } + + foreach ( $this->plugins_required as $plugin => $data ) { + + if ( ! $this->is_plugin_activated( $plugin ) || $this->is_plugin_outdated( $plugin ) ) { + add_action( 'admin_notices', array( $this, 'dependencies_notice' ) ); + $this->loadable = false; + } + } + } + + public function get_plugin_version( $plugin_slug ) { + return get_option( $plugin_slug . '_version' ); + } + + public function is_plugin_outdated( $plugin ) { + $required = ( isset( $this->plugins_required[ $plugin ] ) ? $this->plugins_required[ $plugin ] : false ); + + if ( ! $required ) { + return false; + } + + if ( version_compare( $this->get_plugin_version( $required['version_prefix'] ), $required['version'], '<' ) ) { + return true; + } + + return false; + } + + public function is_plugin_activated( $plugin ) { + + if ( strpos( $plugin, '.php' ) === false ) { + $plugin = trailingslashit( $plugin ) . $plugin . '.php'; + } + + return in_array( $plugin, $this->plugins, true ) || array_key_exists( $plugin, $this->plugins ); + } + + /** + * This method removes accuration from $ver2 if this version is more accurate than $main_ver + */ + public function compare_versions( $main_ver, $ver2, $operator ) { + + $expl_main_ver = explode( '.', $main_ver ); + $expl_ver2 = explode( '.', $ver2 ); + + // Check if ver2 string is more accurate than main_ver + if ( 2 === count( $expl_main_ver ) && count( $expl_ver2 ) > 2 ) { + $new_ver_2 = array_slice( $expl_ver2, 0, 2 ); + $ver2 = implode( '.', $new_ver_2 ); + } + + return version_compare( $main_ver, $ver2, $operator ); + } + + /** + * Checks if WooCommerce is activated + * + * @return boolean true if WooCommerce is activated + */ + public function is_woocommerce_activated() { + return $this->is_plugin_activated( 'woocommerce/woocommerce.php' ); + } + + public function is_wpml_activated() { + return ( $this->is_plugin_activated( 'sitepress-multilingual-cms/sitepress.php' ) && $this->is_plugin_activated( 'woocommerce-multilingual/wpml-woocommerce.php' ) ); + } + + public function woocommerce_version_supports_crud() { + return ( $this->compare_versions( $this->get_plugin_version( 'woocommerce' ), '2.7', '>=' ) ); + } + + public function is_loadable() { + return $this->loadable; + } + + public function dependencies_notice() { + global $dependencies; + $dependencies = $this; + + include_once 'admin/views/html-notice-dependencies.php'; + } + +} + +WC_TS_Dependencies::instance(); diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-ts-install.php b/packages/woocommerce-trusted-shops/includes/class-wc-ts-install.php new file mode 100644 index 000000000..dfdd387d6 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-ts-install.php @@ -0,0 +1,171 @@ + 'updates/woocommerce-ts-update-3.0.0.php', + '4.0.6' => 'updates/woocommerce-ts-update-4.0.6.php', + ); + + /** + * Hook in tabs. + */ + public function __construct() { + if ( ! Package::is_integration() ) { + add_action( 'admin_init', array( __CLASS__, 'check_version' ), 10 ); + } + + add_action( 'admin_init', array( __CLASS__, 'install_actions' ), 5 ); + } + + /** + * Install actions such as installing pages when a button is clicked. + */ + public static function install_actions() { + if ( ! empty( $_GET['do_update_woocommerce_ts'] ) && current_user_can( 'manage_woocommerce' ) ) { + check_admin_referer( 'wc_ts_db_update', 'wc_ts_db_update_nonce' ); + self::update(); + + // Update complete + delete_option( '_wc_ts_needs_update' ); + } + } + + /** + * check_version function. + * + * @access public + * @return void + */ + public static function check_version() { + if ( ! defined( 'IFRAME_REQUEST' ) && ( get_option( 'woocommerce_trusted_shops_version' ) !== WC_trusted_shops()->version || get_option( 'woocommerce_trusted_shops_db_version' ) !== WC_trusted_shops()->version ) ) { + self::install(); + do_action( 'woocommerce_trusted_shops_updated' ); + } + } + + public static function install_integration() { + self::create_cron_jobs(); + self::update_versions(); + } + + protected static function update_versions() { + // Queue upgrades + $current_version = get_option( 'woocommerce_trusted_shops_version', null ); + $current_db_version = get_option( 'woocommerce_trusted_shops_db_version', null ); + + if ( ! is_null( $current_db_version ) && version_compare( $current_db_version, max( array_keys( self::$db_updates ) ), '<' ) ) { + // Update + update_option( '_wc_ts_needs_update', 1 ); + } else { + self::update_db_version(); + } + + self::update_ts_version(); + + do_action( 'woocommerce_trusted_shops_installed' ); + } + + /** + * Install TS + */ + public static function install() { + // Load Translation for default options + $locale = apply_filters( 'plugin_locale', get_locale(), 'woocommerce-germanized' ); + $mofile = WC_trusted_shops()->plugin_path() . '/i18n/languages/woocommerce-trusted-shops.mo'; + + if ( file_exists( WC_trusted_shops()->plugin_path() . '/i18n/languages/woocommerce-trusted-shops-' . $locale . '.mo' ) ) { + $mofile = WC_trusted_shops()->plugin_path() . '/i18n/languages/woocommerce-trusted-shops-' . $locale . '.mo'; + } + + load_textdomain( 'woocommerce-trusted-shops', $mofile ); + + self::create_options(); + self::create_cron_jobs(); + self::update_versions(); + + // Flush rules after install + flush_rewrite_rules(); + } + + /** + * Update WC version to current + */ + private static function update_ts_version() { + delete_option( 'woocommerce_trusted_shops_version' ); + add_option( 'woocommerce_trusted_shops_version', WC_trusted_shops()->version ); + } + + /** + * Update DB version to current + */ + private static function update_db_version( $version = null ) { + delete_option( 'woocommerce_trusted_shops_db_version' ); + add_option( 'woocommerce_trusted_shops_db_version', is_null( $version ) ? WC_trusted_shops()->version : $version ); + } + + /** + * Handle updates + */ + public static function update() { + $current_db_version = get_option( 'woocommerce_trusted_shops_db_version' ); + + foreach ( self::$db_updates as $version => $updater ) { + if ( version_compare( $current_db_version, $version, '<' ) ) { + include $updater; + self::update_db_version( $version ); + } + } + + self::update_db_version(); + } + + /** + * Create cron jobs (clear them first) + */ + private static function create_cron_jobs() { + // Cron jobs + wp_clear_scheduled_hook( 'woocommerce_gzd_trusted_shops_reviews' ); + wp_schedule_event( time(), 'twicedaily', 'woocommerce_gzd_trusted_shops_reviews' ); + } + + /** + * Default options + * + * Sets up the default options used on the settings page + * + * @access public + */ + private static function create_options() { + // Include settings so that we can run through defaults + $options = apply_filters( 'woocommerce_gzd_installation_default_settings', array() ); + + foreach ( $options as $value ) { + + if ( isset( $value['default'] ) && isset( $value['id'] ) ) { + $autoload = isset( $value['autoload'] ) ? (bool) $value['autoload'] : true; + add_option( $value['id'], $value['default'], '', ( $autoload ? 'yes' : 'no' ) ); + } + } + } + } + +endif; + +return new WC_TS_Install(); diff --git a/packages/woocommerce-trusted-shops/includes/class-wc-ts-settings-handler.php b/packages/woocommerce-trusted-shops/includes/class-wc-ts-settings-handler.php new file mode 100644 index 000000000..d9eb2ef4d --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/class-wc-ts-settings-handler.php @@ -0,0 +1,77 @@ +id = 'trusted-shops'; + $this->label = _x( 'Trusted Shops', 'trusted-shops', 'woocommerce-germanized' ); + + parent::__construct(); + } + + /** + * Get settings array + * + * @return array + */ + public function get_settings() { + $admin = WC_trusted_shops()->trusted_shops->get_dependency( 'admin' ); + $settings = $admin->get_settings(); + + return $settings; + } + + public function get_settings_for_section_core( $section_id ) { + $admin = WC_trusted_shops()->trusted_shops->get_dependency( 'admin' ); + $settings = $admin->get_settings(); + + return $settings; + } + + public function output() { + global $current_section; + + $settings = $this->get_settings_for_section_core( '' ); + $sidebar = $this->get_sidebar(); + + include_once WC_trusted_shops()->plugin_path() . '/includes/admin/views/html-settings-section.php'; + } + + public function get_sidebar() { + $admin = WC_trusted_shops()->trusted_shops->get_dependency( 'admin' ); + $sidebar = $admin->get_sidebar(); + + return $sidebar; + } + + /** + * Save settings + */ + public function save() { + $settings = $this->get_settings_for_section_core( '' ); + + do_action( 'woocommerce_ts_before_save', $settings ); + WC_Admin_Settings::save_fields( $settings ); + do_action( 'woocommerce_ts_after_save', $settings ); + } + } + +endif; + + diff --git a/packages/woocommerce-trusted-shops/includes/compatibility/class-wc-ts-compatibility-wpml-string-translation.php b/packages/woocommerce-trusted-shops/includes/compatibility/class-wc-ts-compatibility-wpml-string-translation.php new file mode 100644 index 000000000..44e531998 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/compatibility/class-wc-ts-compatibility-wpml-string-translation.php @@ -0,0 +1,342 @@ + defined( 'WPML_ST_VERSION' ) ? WPML_ST_VERSION : '1.0', + ) + ); + } + + public function is_activated() { + global $sitepress; + + return defined( 'WPML_ST_VERSION' ) && isset( $sitepress ) ? true : false; + } + + public function load() { + if ( is_admin() ) { + $this->admin_translate_options(); + } + } + + public function get_languages() { + global $sitepress; + $codes = array(); + + if ( isset( $sitepress ) && is_callable( array( $sitepress, 'get_ls_languages' ) ) ) { + $languages = $sitepress->get_ls_languages(); + + if ( ! empty( $languages ) ) { + $codes = array_keys( $languages ); + } + } + + return $codes; + } + + public function get_language_name( $language = '' ) { + global $sitepress; + + if ( empty( $language ) ) { + $language = $this->get_current_language(); + } + + return $sitepress->get_display_language_name( $language ); + } + + public function get_current_language() { + global $sitepress; + + return $sitepress->get_current_language(); + } + + public function get_default_language() { + global $sitepress; + $default = ''; + + if ( isset( $sitepress ) && is_callable( array( $sitepress, 'get_default_language' ) ) ) { + $default = $sitepress->get_default_language(); + } + + return $default; + } + + /** + * Register strings that are translatable within the settings panel. Strings will be loaded in the admin user's language + * within the settings screen only. + * + * @return array + */ + public function get_translatable_options() { + return apply_filters( 'woocommerce_gzd_wpml_translatable_options', array() ); + } + + /** + * By default WPML allow only certain strings to be translated within the administration area (e.g. blog title). + * If you want some translatable strings to be loaded globally within the admin panel use the filter accordingly. + * + * @return array + */ + public function get_translatable_admin_options() { + return apply_filters( 'woocommerce_gzd_wpml_translatable_admin_options', array() ); + } + + public function admin_translate_options() { + $this->set_filters(); + } + + public function set_filters() { + $admin_strings = $this->get_translatable_admin_options(); + + if ( $this->enable_option_filters() ) { + + foreach ( $this->get_translatable_options() as $option => $args ) { + add_filter( 'option_' . $option, array( $this, 'translate_option_filter' ), 10, 2 ); + add_filter( 'pre_update_option_' . $option, array( $this, 'pre_update_translation_filter' ), 10, 3 ); + + wc_ts_remove_class_filter( 'option_' . $option, 'WPML_Admin_Texts', 'icl_st_translate_admin_string', 10 ); + } + } elseif ( ! empty( $admin_strings ) ) { + + foreach ( $admin_strings as $option => $args ) { + add_filter( 'option_' . $option, array( $this, 'translate_option_filter' ), 10, 2 ); + wc_ts_remove_class_filter( 'option_' . $option, 'WPML_Admin_Texts', 'icl_st_translate_admin_string', 10 ); + } + } + } + + protected function enable_option_filters() { + $enable = false; + + if ( isset( $_GET['tab'] ) && ( 'trusted-shops' === $_GET['tab'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended + $enable = true; + } + + return apply_filters( 'woocommerce_gzd_enable_wpml_string_translation_settings_filters', $enable ); + } + + public function get_string_language( $string_id, $option = '' ) { + if ( $string = $this->get_string_by_id( $string_id ) ) { + return $string->language; + } + + global $WPML_String_Translation; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.VariableNotSnakeCase + + return $WPML_String_Translation->get_current_string_language( $option ); // phpcs:ignore WordPress.NamingConventions.ValidVariableName.VariableNotSnakeCase + } + + public function get_string_by_id( $string_id ) { + global $wpdb; + + $string = $wpdb->get_results( $wpdb->prepare( "SELECT * FROM {$wpdb->prefix}icl_strings WHERE id=%d LIMIT 1", $string_id ) ); + + if ( $string ) { + return $string[0]; + } + + return false; + } + + public function get_string_id( $option, $context = '' ) { + $context = empty( $context ) ? 'admin_texts_' . $option : $context; + + return icl_st_is_registered_string( $context, $option ); + } + + public function get_string_value( $string_id ) { + if ( $string = $this->get_string_by_id( $string_id ) ) { + return $string->value; + } + + return false; + } + + public function update_string_value( $string_id, $value ) { + global $wpdb; + + $value = maybe_serialize( $value ); + + $wpdb->update( "{$wpdb->prefix}icl_strings", array( 'value' => $value ), array( 'id' => $string_id ) ); + } + + public function delete_string_translation( $string_id, $language ) { + global $wpdb; + + $wpdb->query( $wpdb->prepare( "DELETE FROM {$wpdb->prefix}icl_string_translations WHERE string_id=%d AND language=%s", $string_id, $language ) ); + } + + public function update_string_translation( $option, $language, $value, $status = '' ) { + if ( empty( $status ) ) { + $status = ICL_TM_COMPLETE; + } + + if ( $string_id = $this->get_string_id( $option ) ) { + icl_add_string_translation( $string_id, $language, $value, $status ); + } + + icl_update_string_translation( $option, $language, $value, $status ); + + // Make sure that the string is stored within the WPML translatable option names + $option_names = get_option( '_icl_admin_option_names', array() ); + $option_names[ $option ] = 1; + + update_option( '_icl_admin_option_names', $option_names ); + } + + public function get_string_translation( $string_id, $language, $status = '' ) { + $translations = icl_get_string_translations_by_id( $string_id ); + $status = empty( $status ) ? ICL_TM_COMPLETE : $status; + + if ( isset( $translations[ $language ] ) && $translations[ $language ]['status'] === $status ) { + return $translations[ $language ]['value']; + } + + return false; + } + + public function get_translated_string( $option, $language, $context = '' ) { + $value = null; + + if ( $string_id = $this->get_string_id( $option, $context ) ) { + $value = $this->get_string_translation( $string_id, $language ); + } + + return $value; + } + + public function register_string( $option, $value, $context = '' ) { + $context = empty( $context ) ? 'admin_texts_' . $option : $context; + + return icl_register_string( $context, $option, $value ); + } + + public function pre_update_translation_filter( $new_value, $old_value, $option ) { + $string_options = $this->get_translatable_options(); + + if ( is_array( $new_value ) || is_array( $string_options[ $option ] ) ) { + + if ( ! is_array( $new_value ) ) { + $new_value = array(); + } + + $args = $string_options[ $option ]; + + foreach ( $new_value as $id => $options ) { + foreach ( $options as $key => $value ) { + if ( in_array( $key, $args ) ) { // phpcs:ignore WordPress.PHP.StrictInArray.MissingTrueStrict + $old_value_internal = isset( $old_value[ $id ][ $key ] ) ? $old_value[ $id ][ $key ] : ''; + $new_value[ $id ][ $key ] = $this->pre_update_translation( $value, $old_value_internal, "[{$option}][{$id}]{$key}", "admin_texts_{$option}" ); + } + } + } + + return $new_value; + } else { + return $this->pre_update_translation( $new_value, $old_value, $option ); + } + } + + protected function pre_update_translation( $new_value, $old_value, $option, $context = '' ) { + $org_string_id = $this->get_string_id( $option, $context ); + $strings_language = $this->get_string_language( $org_string_id, $option ); + $return_value = $old_value; + + if ( $strings_language === $this->get_current_language() || 'all' === $this->get_current_language() ) { + + $current_string_value = $this->get_string_value( $org_string_id ); + + // Update original string value + if ( $org_string_id && ( $new_value !== $old_value || $current_string_value !== $new_value ) ) { + $this->update_string_value( $org_string_id, $new_value ); + } + + $return_value = $new_value; + + } else { + $update_translation = true; + + if ( $org_string_id ) { + $org_string = $this->get_string_value( $org_string_id ); + + /** + * Remove translation if it equals original string + * Use woocommerce_gzd_wpml_remove_translation_empty_equal filter to disallow string deletion which results in "real" option translations + */ + if ( ( $org_string === $new_value || empty( $new_value ) ) && apply_filters( 'woocommerce_gzd_wpml_remove_translation_empty_equal', true, $option, $new_value, $old_value ) ) { + $this->delete_string_translation( $org_string_id, $this->get_current_language() ); + + $return_value = $old_value; + $update_translation = false; + } + } + + if ( $update_translation ) { + $this->update_string_translation( $option, $this->get_current_language(), $new_value ); + } + } + + // Allow WPML to delete the cache + do_action( "update_option_{$option}", $old_value, $return_value, $option ); + + return $return_value; + } + + public function translate_option_filter( $org_value, $option ) { + $string_options = $this->get_translatable_options(); + + if ( is_array( $org_value ) || is_array( $string_options[ $option ] ) ) { + + if ( ! is_array( $org_value ) ) { + $org_value = array(); + } + + $args = $string_options[ $option ]; + + foreach ( $org_value as $id => $options ) { + foreach ( $options as $key => $value ) { + if ( in_array( $key, $args ) ) { // phpcs:ignore WordPress.PHP.StrictInArray.MissingTrueStrict + $org_value[ $id ][ $key ] = $this->translate_option( $value, "[{$option}][{$id}]{$key}", "admin_texts_{$option}" ); + } + } + } + + return $org_value; + } else { + return $this->translate_option( $org_value, $option ); + } + } + + protected function translate_option( $org_value, $option, $context = '' ) { + $string_id = $this->get_string_id( $option, $context ); + $language = $this->get_current_language(); + + if ( ! $string_id ) { + $string_id = $this->register_string( $option, $org_value, $context ); + } + + $translation = $this->get_string_translation( $string_id, $language ); + + if ( false !== $translation ) { + $org_value = $translation; + } + + return $org_value; + } + + public function pre_update_translate_checkboxes( $new_value, $old_value, $option ) { + return $new_value; + } +} diff --git a/packages/woocommerce-trusted-shops/includes/emails/class-wc-ts-email-customer-trusted-shops.php b/packages/woocommerce-trusted-shops/includes/emails/class-wc-ts-email-customer-trusted-shops.php new file mode 100644 index 000000000..1bbf41b38 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/emails/class-wc-ts-email-customer-trusted-shops.php @@ -0,0 +1,165 @@ +id = 'customer_trusted_shops'; + $this->title = _x( 'Trusted Shops Review Reminder', 'trusted-shops', 'woocommerce-germanized' ); + $this->description = _x( 'This E-Mail is being sent to a customer to remind him about the possibility to leave a review at Trusted Shops.', 'trusted-shops', 'woocommerce-germanized' ); + + $this->template_html = 'emails/customer-trusted-shops.php'; + $this->template_plain = 'emails/plain/customer-trusted-shops.php'; + $this->helper = function_exists( 'wc_gzd_get_email_helper' ) ? wc_gzd_get_email_helper( $this ) : false; + + // Triggers for this email + add_action( 'woocommerce_germanized_trusted_shops_review_notification', array( $this, 'trigger' ) ); + + $this->placeholders = array( + '{site_title}' => $this->get_blogname(), + '{order_number}' => '', + '{order_date}' => '', + ); + + // Call parent constuctor + parent::__construct(); + + $this->customer_email = true; + } + + /** + * Get email subject. + * + * @since 3.1.0 + * @return string + */ + public function get_default_subject() { + return _x( 'Please rate your {site_title} order from {order_date}', 'trusted-shops', 'woocommerce-germanized' ); + } + + /** + * Get email heading. + * + * @since 3.1.0 + * @return string + */ + public function get_default_heading() { + return _x( 'Please rate your Order', 'trusted-shops', 'woocommerce-germanized' ); + } + + /** + * trigger function. + * + * @access public + * @return void + */ + public function trigger( $order_id ) { + if ( $this->helper ) { + $this->helper->setup_locale(); + } else { + $this->setup_locale(); + } + + if ( $order_id ) { + $this->object = wc_get_order( $order_id ); + $this->recipient = wc_ts_get_crud_data( $this->object, 'billing_email' ); + + $this->placeholders['{order_date}'] = wc_gzd_get_order_date( $this->object, wc_date_format() ); + $this->placeholders['{order_number}'] = $this->object->get_order_number(); + } + + if ( $this->helper ) { + $this->helper->setup_email_locale(); + } + + if ( $this->is_enabled() && $this->get_recipient() ) { + $this->send( $this->get_recipient(), $this->get_subject(), $this->get_content(), $this->get_headers(), $this->get_attachments() ); + } + + if ( $this->helper ) { + $this->helper->restore_email_locale(); + $this->helper->restore_locale(); + } else { + $this->restore_locale(); + } + } + + /** + * Return content from the additional_content field. + * + * Displayed above the footer. + * + * @since 2.0.4 + * @return string + */ + public function get_additional_content() { + if ( is_callable( 'parent::get_additional_content' ) ) { + return parent::get_additional_content(); + } + + return ''; + } + + /** + * get_content_html function. + * + * @access public + * @return string + */ + public function get_content_html() { + return wc_get_template_html( + $this->template_html, + array( + 'order' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => false, + 'email' => $this, + ) + ); + } + + /** + * Get content plain. + * + * @return string + */ + public function get_content_plain() { + return wc_get_template_html( + $this->template_plain, + array( + 'order' => $this->object, + 'email_heading' => $this->get_heading(), + 'additional_content' => $this->get_additional_content(), + 'sent_to_admin' => false, + 'plain_text' => false, + 'email' => $this, + ) + ); + } + } + +endif; + +return new WC_TS_Email_Customer_Trusted_Shops(); diff --git a/packages/woocommerce-trusted-shops/includes/updates/woocommerce-ts-update-3.0.0.php b/packages/woocommerce-trusted-shops/includes/updates/woocommerce-ts-update-3.0.0.php new file mode 100644 index 000000000..c36776453 --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/updates/woocommerce-ts-update-3.0.0.php @@ -0,0 +1,23 @@ +get_wc_product(); + } + + $value = null; + + $getter = substr( $key, 0, 3 ) === 'get' ? $key : "get_$key"; + $key = substr( $key, 0, 3 ) === 'get' ? substr( $key, 3 ) : $key; + + if ( 'id' === $key && is_callable( array( $object, 'is_type' ) ) && $object->is_type( 'variation' ) && ! wc_ts_woocommerce_supports_crud() ) { + $key = 'variation_id'; + } elseif ( 'parent' === $key && is_callable( array( $object, 'is_type' ) ) && $object->is_type( 'variation' ) && ! wc_ts_woocommerce_supports_crud() ) { + // Set getter to parent so that it is not being used for pre 2.7 + $key = 'id'; + $getter = 'parent'; + } + + $getter_mapping = array( + 'parent' => 'get_parent_id', + 'completed_date' => 'get_date_completed', + 'order_date' => 'get_date_created', + 'product_type' => 'get_type', + 'order_type' => 'get_type', + ); + + if ( array_key_exists( $key, $getter_mapping ) ) { + $getter = $getter_mapping[ $key ]; + } + + if ( is_callable( array( $object, $getter ) ) ) { + $reflection = new ReflectionMethod( $object, $getter ); + if ( $reflection->isPublic() ) { + $value = $object->{$getter}(); + } + } elseif ( wc_ts_woocommerce_supports_crud() ) { + // Prefix meta if suppress_suffix is not set + if ( substr( $key, 0, 1 ) !== '_' && ! $suppress_suffix ) { + $key = '_' . $key; + } + + $value = $object->get_meta( $key ); + } else { + $key = substr( $key, 0, 1 ) === '_' ? substr( $key, 1 ) : $key; + $value = $object->{$key}; + } + + return $value; + } +} + +if ( ! function_exists( 'wc_ts_woocommerce_supports_crud' ) ) { + + function wc_ts_woocommerce_supports_crud() { + return WC_TS_Dependencies::instance()->woocommerce_version_supports_crud(); + } +} + +if ( ! function_exists( 'wc_ts_help_tip' ) ) { + + function wc_ts_help_tip( $tip, $allow_html = false ) { + if ( function_exists( 'wc_help_tip' ) ) { + return wc_help_tip( $tip, $allow_html ); + } + + return '[?]'; + } +} + +if ( ! function_exists( 'wc_ts_set_crud_data' ) ) { + + function wc_ts_set_crud_data( $object, $key, $value ) { + if ( wc_ts_woocommerce_supports_crud() ) { + + $key_unprefixed = substr( $key, 0, 1 ) === '_' ? substr( $key, 1 ) : $key; + $setter = substr( $key_unprefixed, 0, 3 ) === 'set' ? $key : "set_{$key_unprefixed}"; + + if ( is_callable( array( $object, $setter ) ) ) { + $reflection = new ReflectionMethod( $object, $setter ); + if ( $reflection->isPublic() ) { + $object->{$setter}( $value ); + } + } else { + $object = wc_ts_set_crud_meta_data( $object, $key, $value ); + } + } else { + $object = wc_ts_set_crud_meta_data( $object, $key, $value ); + } + return $object; + } +} + +if ( ! function_exists( 'wc_ts_set_crud_meta_data' ) ) { + function wc_ts_set_crud_meta_data( $object, $key, $value ) { + + if ( wc_ts_woocommerce_supports_crud() ) { + $object->update_meta_data( $key, $value ); + } else { + update_post_meta( wc_ts_get_crud_data( $object, 'id' ), $key, $value ); + } + return $object; + } +} + +if ( ! function_exists( 'wc_ts_get_order_date' ) ) { + + function wc_ts_get_order_date( $order, $format = '' ) { + $date_formatted = ''; + + if ( function_exists( 'wc_format_datetime' ) ) { + return wc_format_datetime( $order->get_date_created(), $format ); + } else { + $date = $order->order_date; + } + + if ( empty( $format ) ) { + $format = get_option( 'date_format' ); + } + + if ( ! empty( $date ) ) { + $date_formatted = date_i18n( $format, strtotime( $date ) ); + } + + return $date_formatted; + } +} + +if ( ! function_exists( 'wc_ts_get_order_currency' ) ) { + + function wc_ts_get_order_currency( $order ) { + if ( wc_ts_woocommerce_supports_crud() ) { + return $order->get_currency(); + } + + return $order->get_order_currency(); + } +} + +if ( ! function_exists( 'wc_ts_get_order_language' ) ) { + + function wc_ts_get_order_language( $order ) { + $order_id = is_numeric( $order ) ? $order : wc_ts_get_crud_data( $order, 'id' ); + + return get_post_meta( $order_id, 'wpml_language', true ); + } +} + +if ( ! function_exists( 'wc_ts_switch_language' ) ) { + + function wc_ts_switch_language( $lang, $set_default = false ) { + global $sitepress; + global $wc_ts_original_lang; + + if ( $set_default ) { + $wc_ts_original_lang = $lang; + } + + if ( isset( $sitepress ) && is_callable( array( $sitepress, 'get_current_language' ) ) && is_callable( array( $sitepress, 'switch_lang' ) ) ) { + if ( $sitepress->get_current_language() !== $lang ) { + + $sitepress->switch_lang( $lang, true ); + + // Somehow WPML doesn't automatically change the locale + if ( is_callable( array( $sitepress, 'reset_locale_utils_cache' ) ) ) { + $sitepress->reset_locale_utils_cache(); + } + + if ( function_exists( 'switch_to_locale' ) ) { + switch_to_locale( get_locale() ); + + // Filter on plugin_locale so load_plugin_textdomain loads the correct locale. + add_filter( 'plugin_locale', 'get_locale' ); + + // Init WC locale. + WC()->load_plugin_textdomain(); + WC_trusted_shops()->load_plugin_textdomain(); + WC_trusted_shops()->trusted_shops->refresh(); + } + + do_action( 'woocommerce_gzd_trusted_shops_switched_language', $lang, $wc_ts_original_lang ); + } + } + + do_action( 'woocommerce_gzd_trusted_shops_switch_language', $lang, $wc_ts_original_lang ); + } +} + +if ( ! function_exists( 'wc_ts_restore_language' ) ) { + + function wc_ts_restore_language() { + global $wc_ts_original_lang; + + if ( isset( $wc_ts_original_lang ) && ! empty( $wc_ts_original_lang ) ) { + wc_ts_switch_language( $wc_ts_original_lang ); + } + } +} + +if ( ! function_exists( 'wc_ts_remove_class_filter' ) ) { + /** + * Remove Class Filter Without Access to Class Object + * + * In order to use the core WordPress remove_filter() on a filter added with the callback + * to a class, you either have to have access to that class object, or it has to be a call + * to a static method. This method allows you to remove filters with a callback to a class + * you don't have access to. + * + * Works with WordPress 1.2+ (4.7+ support added 9-19-2016) + * Updated 2-27-2017 to use internal WordPress removal for 4.7+ (to prevent PHP warnings output) + * + * @param string $tag Filter to remove + * @param string $class_name Class name for the filter's callback + * @param string $method_name Method name for the filter's callback + * @param int $priority Priority of the filter (default 10) + * + * @return bool Whether the function is removed. + */ + function wc_ts_remove_class_filter( $tag, $class_name = '', $method_name = '', $priority = 10 ) { + global $wp_filter; + + // Check that filter actually exists first + if ( ! isset( $wp_filter[ $tag ] ) ) { + return false; + } + + /** + * If filter config is an object, means we're using WordPress 4.7+ and the config is no longer + * a simple array, rather it is an object that implements the ArrayAccess interface. + * + * To be backwards compatible, we set $callbacks equal to the correct array as a reference (so $wp_filter is updated) + * + * @see https://make.wordpress.org/core/2016/09/08/wp_hook-next-generation-actions-and-filters/ + */ + if ( is_object( $wp_filter[ $tag ] ) && isset( $wp_filter[ $tag ]->callbacks ) ) { + // Create $fob object from filter tag, to use below + $fob = $wp_filter[ $tag ]; + $callbacks = &$wp_filter[ $tag ]->callbacks; + } else { + $callbacks = &$wp_filter[ $tag ]; + } + + // Exit if there aren't any callbacks for specified priority + if ( ! isset( $callbacks[ $priority ] ) || empty( $callbacks[ $priority ] ) ) { + return false; + } + + // Loop through each filter for the specified priority, looking for our class & method + foreach ( (array) $callbacks[ $priority ] as $filter_id => $filter ) { + + // Filter should always be an array - array( $this, 'method' ), if not goto next + if ( ! isset( $filter['function'] ) || ! is_array( $filter['function'] ) ) { + continue; + } + + // If first value in array is not an object, it can't be a class + if ( ! is_object( $filter['function'][0] ) ) { + continue; + } + + // Method doesn't match the one we're looking for, goto next + if ( $filter['function'][1] !== $method_name ) { + continue; + } + + // Method matched, now let's check the Class + if ( get_class( $filter['function'][0] ) === $class_name ) { + + // WordPress 4.7+ use core remove_filter() since we found the class object + if ( isset( $fob ) ) { + // Handles removing filter, reseting callback priority keys mid-iteration, etc. + $fob->remove_filter( $tag, $filter['function'], $priority ); + + } else { + // Use legacy removal process (pre 4.7) + unset( $callbacks[ $priority ][ $filter_id ] ); + // and if it was the only filter in that priority, unset that priority + if ( empty( $callbacks[ $priority ] ) ) { + unset( $callbacks[ $priority ] ); + } + // and if the only filter for that tag, set the tag to an empty array + if ( empty( $callbacks ) ) { + $callbacks = array(); + } + // Remove this filter from merged_filters, which specifies if filters have been sorted + unset( $GLOBALS['merged_filters'][ $tag ] ); + } + + return true; + } + } + + return false; + } +} + +if ( ! function_exists( 'wc_ts_remove_class_action' ) ) { + /** + * Remove Class Action Without Access to Class Object + * + * In order to use the core WordPress remove_action() on an action added with the callback + * to a class, you either have to have access to that class object, or it has to be a call + * to a static method. This method allows you to remove actions with a callback to a class + * you don't have access to. + * + * Works with WordPress 1.2+ (4.7+ support added 9-19-2016) + * + * @param string $tag Action to remove + * @param string $class_name Class name for the action's callback + * @param string $method_name Method name for the action's callback + * @param int $priority Priority of the action (default 10) + * + * @return bool Whether the function is removed. + */ + function wc_ts_remove_class_action( $tag, $class_name = '', $method_name = '', $priority = 10 ) { + wc_ts_remove_class_filter( $tag, $class_name, $method_name, $priority ); + } +} diff --git a/packages/woocommerce-trusted-shops/includes/widgets/class-wc-trusted-shops-widget-review-sticker.php b/packages/woocommerce-trusted-shops/includes/widgets/class-wc-trusted-shops-widget-review-sticker.php new file mode 100644 index 000000000..6a39d08ce --- /dev/null +++ b/packages/woocommerce-trusted-shops/includes/widgets/class-wc-trusted-shops-widget-review-sticker.php @@ -0,0 +1,66 @@ +widget_cssclass = 'woocommerce woocommerce_gzd widget_trusted_shops_review_sticker'; + $this->widget_description = _x( 'Show your TS shop review sticker.', 'trusted-shops', 'woocommerce-germanized' ); + $this->widget_id = 'woocommerce_gzd_widget_trusted_shops_shop_review_sticker'; + $this->widget_name = _x( 'Trusted Shops Shop Review Sticker', 'trusted-shops', 'woocommerce-germanized' ); + $this->settings = array( + 'title' => array( + 'type' => 'text', + 'std' => _x( 'Trusted Shops Reviews', 'trusted-shops', 'woocommerce-germanized' ), + 'label' => _x( 'Title', 'trusted-shops', 'woocommerce-germanized' ), + ), + ); + parent::__construct(); + } + + /** + * widget function. + * + * @see WP_Widget + * @access public + * @param array $args + * @param array $instance + * @return void + */ + public function widget( $args, $instance ) { + extract( $args ); // phpcs:ignore WordPress.PHP.DontExtract.extract_extract + + $element = "#ts_review_sticker_{$this->number}"; + + $title = apply_filters( 'widget_title', empty( $instance['title'] ) ? _x( 'Trusted Shops Reviews', 'trusted-shops', 'woocommerce-germanized' ) : $instance['title'], $instance, $this->id_base ); + + echo wp_kses_post( $before_widget ); + + if ( $title ) { + echo wp_kses_post( $before_title . $title . $after_title ); + } + + echo '
'; + + echo do_shortcode( '[trusted_shops_review_sticker element="' . $element . '"]' ); + + echo '
'; + + echo wp_kses_post( $after_widget ); + } +} + + diff --git a/packages/woocommerce-trusted-shops/license.txt b/packages/woocommerce-trusted-shops/license.txt new file mode 100644 index 000000000..e4e3f0c5f --- /dev/null +++ b/packages/woocommerce-trusted-shops/license.txt @@ -0,0 +1,699 @@ +WooCommerce Trusted Shops + +Copyright 2011 by the contributors + +This program is free software; you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation; either version 3 of the License, or +(at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with this program; if not, write to the Free Software +Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + +This program incorporates work covered by the following copyright and +permission notices: + + WooCommerce + +=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= + + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright © 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright © + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright © + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. \ No newline at end of file diff --git a/packages/woocommerce-trusted-shops/readme.txt b/packages/woocommerce-trusted-shops/readme.txt new file mode 100644 index 000000000..cf8e9edd9 --- /dev/null +++ b/packages/woocommerce-trusted-shops/readme.txt @@ -0,0 +1,152 @@ +=== Trustbadge Reviews for WooCommerce === +Contributors: vendidero, trustbadge +Tags: advanced reviews, badge, best reviews, business ratings, business reviews, confirm email reviews, google rating, google shopping, product ratings, product reviews, rate products, rating summary, Rating Widget, ratings, reputation, review widget, review, reviews easy, reviews, rich snippets, seal, seo, star rating, stars, trust, trustbadge, trusted reviews, trusted shops, ts, user rating, user reviews, woocommerce trusted shops, woocommerce +Donate link: http://www.trustbadge.com +Requires at least: 4.9 +Tested up to: 5.9 +WC requires at least: 3.4 +WC tested up to: 6.1 +Stable tag: 4.0.15 +Requires PHP: 5.6 +License: GPLv3 +License URI: http://www.gnu.org/licenses/gpl-3.0.html + +Show that your customers love you with reviews in your online store and boost your business with the free Trustbadge Reviews Plugin for WooCommerce. +== Description == + += Building trust - in just 5 minutes! = + +The well-known Trustmark, the Buyer Protection and the authentic reviews from Trusted Shops have stood for trust for over 20 years. More than 30,000 online shops throughout Europe use our Trust solutions for more traffic, higher sales and better conversion rates. + +Trustbadge Reviews for WooCommerce is the easiest and fastest way to convince visitors of the trustworthiness of your online shop. The simple installation guarantees product use in just 5 minutes and usually requires little to no prior technical knowledge. With our extension you are always technically up to date and have no additional maintenance effort. + +Your benefit: With just a few clicks, visitors to your online shop can see trust elements such as the Trustbadge or other on-site widgets, can benefit from buyer protection and are automatically asked for feedback after placing an order. + += All features at a glance: = + +* Show Trustbadge, integrate Buyer Protection & collect shop reviews +* Show Shop Review Sticker with rating comments +* Collect and display Product Reviews +* Configure multi-shops with multiple Trusted Shops IDs + +Please note: To use the extension Trustbadge Reviews for WooCommerce, you need an existing Trusted Shops membership. You can find out more about the products and benefits of Trusted Shops on our [website](https://business.trustedshops.com/) or by calling: +44 23364 5906 + +== Installation == + += Minimal Requirements = + +* WordPress 4.9 or newer +* WooCommerce 3.0 +* PHP Version 5.6 or newer + += Shortcodes = + +`[trusted_shops_rich_snippets]` +Use this shortcode to embed your updated Trusted Shops Rich Snippets within your post/page or product description. + +`[trusted_shops_reviews]` +Embed your Trusted Shops Review Image within your post/page or product description. + +`[trusted_shops_badge]` +Embed your Trusted Shops Badge within your content. + +== Frequently Asked Questions == + += Do you need help with Trustbadge Reviews for WooCommerce? = +A detailed integration manual can be found in our [Help Centre](https://help.etrusted.com/hc/en-gb/articles/360046269991-Using-Trusted-Shops-with-WooCommerce). + += How can I become a Trusted Shops Member? = +You can find out more about the products and benefits of Trusted Shops on our [website](https://business.trustedshops.com/) or by calling: +44 23364 5906 + +== Screenshots == + +1. Screenshot 1 +2. Screenshot 2 +3. Screenshot 3 +4. Screenshot 4 +5. Screenshot 5 + +== Changelog == += 4.0.15 = +* Improvement: PHP Code Sniffer fixes + += 4.0.14 = +* Improvement: Harden URL escaping + += 4.0.13 = +* Improvement: Force parent GTIN/MPN within trusted shops wrapper + += 4.0.12 = +* Fix: Custom selectors defaults +* Improvement: Updating default settings when switching to standard mode + += 4.0.11 = +* Improvement: CSV export format +* Improvement: WP 5.8, Woo 5.5 support + += 4.0.8 = +* Improvement: Use Woo payment method title +* Improvement: Better function exists checking + += 4.0.4 = +* Improvement: Indicate Woo 4.0 + WP 5.4 support +* Improvement: Removed legacy support +* Fix: Email schedule +* Fix: Force review widget template override + += 4.0.0 = +* Improvement: Legacy code removals +* Improvement: PHP 5.6 +* Improvement: WC 3.8 support + += 3.0.3 = +* Improvement: Indicate correct module name in templates for suppport purposes + += 3.0.2 = +* Improvement: Do only replace Woo reviews when product sticker is enabled + += 3.0.1 = +* Fix: Autoloading class + += 3.0.0 = +* Feature - Better WPML support +* Feature - Better setting control +* Improvement - Code refactoring + += 2.2.0 = +* Feature - WC 3.2 Compatibility +* Fix - Review stars not showing on product page + += 2.0.1 = +* Feature - WC 2.7 beta compatibility +* Feature - Rich snippets image +* Feature - Brand/MPN attribute +* Fix - Template globals filter removal +* Fix - Settings show/hide + += 2.0.3 = +* Fix - Better Review Widget Update + += 2.0.2 = +* Fix - Star Size Option + += 2.0.1 = +* Fix - Review Collector Export + += 2.0.0 = +* Feature - Product Reviews +* Feature - Product Review Sticker/Stars +* Feature - GTIN for Product Reviews +* Feature - In Standard Mode no need to insert JS Trustbadge Code +* Feature - Expert Mode for code adjustments + += 1.1.0 = +* Feature - Trusted Shops Review Collector + += 1.0.0 = +* Feature - Trusted Shops Integration + +== Upgrade Notice == + += 1.0.0 = +no upgrade - just install :) diff --git a/packages/woocommerce-trusted-shops/src/Package.php b/packages/woocommerce-trusted-shops/src/Package.php new file mode 100644 index 000000000..77bd2c54d --- /dev/null +++ b/packages/woocommerce-trusted-shops/src/Package.php @@ -0,0 +1,113 @@ + +
+

+
+ version, '3.1', '>=' ) ? true : false; + } + + public static function is_integration() { + return class_exists( 'WooCommerce_Germanized' ) ? true : false; + } + + private static function includes() { + include_once self::get_path() . '/includes/class-wc-trusted-shops-core.php'; + } + + public static function init_hooks() {} + + /** + * Return the version of the package. + * + * @return string + */ + public static function get_version() { + return self::VERSION; + } + + /** + * Return the path to the package. + * + * @return string + */ + public static function get_path() { + return dirname( __DIR__ ); + } + + /** + * Return the path to the package. + * + * @return string + */ + public static function get_url() { + return plugins_url( '', __DIR__ ); + } + + public static function get_assets_url() { + return self::get_url() . '/assets'; + } + + private static function define_constant( $name, $value ) { + if ( ! defined( $name ) ) { + define( $name, $value ); + } + } +} diff --git a/packages/woocommerce-trusted-shops/templates/emails/cancel-review-reminder.php b/packages/woocommerce-trusted-shops/templates/emails/cancel-review-reminder.php new file mode 100644 index 000000000..55f6099f5 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/emails/cancel-review-reminder.php @@ -0,0 +1,16 @@ + + +
+

' . esc_html_x( 'cancel review reminder', 'trusted-shops', 'woocommerce-germanized' ) . '' ) ); ?>

+
diff --git a/packages/woocommerce-trusted-shops/templates/emails/customer-trusted-shops.php b/packages/woocommerce-trusted-shops/templates/emails/customer-trusted-shops.php new file mode 100644 index 000000000..fb1674662 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/emails/customer-trusted-shops.php @@ -0,0 +1,37 @@ + + + + +

+

+ + + + + +
+ + + + diff --git a/packages/woocommerce-trusted-shops/templates/emails/plain/customer-trusted-shops.php b/packages/woocommerce-trusted-shops/templates/emails/plain/customer-trusted-shops.php new file mode 100644 index 000000000..f8666ed09 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/emails/plain/customer-trusted-shops.php @@ -0,0 +1,34 @@ +trusted_shops->get_new_review_link( wc_ts_get_crud_data( $order, 'billing_email' ), $order->get_order_number() ) ) . "\n\n"; + +echo "\n\n----------------------------------------\n\n"; + +/** + * Show user-defined additional content - this is set in each email's settings. + */ +if ( $additional_content ) { + echo esc_html( wp_strip_all_tags( wptexturize( $additional_content ) ) ); + echo "\n\n----------------------------------------\n\n"; +} + +echo wp_kses_post( apply_filters( 'woocommerce_email_footer_text', get_option( 'woocommerce_email_footer_text' ) ) ); diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/product-sticker-tpl.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-sticker-tpl.php new file mode 100644 index 000000000..c9635f19e --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-sticker-tpl.php @@ -0,0 +1,34 @@ + + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/product-sticker.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-sticker.php new file mode 100644 index 000000000..b7a3517d4 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-sticker.php @@ -0,0 +1,21 @@ +get_product_skus( $post->ID ); +?> + +
get_selector( 'product_sticker' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>>
+ + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/product-widget-tpl.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-widget-tpl.php new file mode 100644 index 000000000..1e27f02eb --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-widget-tpl.php @@ -0,0 +1,22 @@ + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/product-widget.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-widget.php new file mode 100644 index 000000000..6e3db98ef --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/product-widget.php @@ -0,0 +1,25 @@ +ID ); +$plugin = isset( $plugin ) ? $plugin : WC_trusted_shops()->trusted_shops; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited +$skus = $plugin->get_product_skus( $post->ID ); +?> + +
get_selector( 'product_widget' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>>
+ + + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/review-sticker-tpl.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/review-sticker-tpl.php new file mode 100644 index 000000000..0ac4e3230 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/review-sticker-tpl.php @@ -0,0 +1,33 @@ + + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/review-sticker.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/review-sticker.php new file mode 100644 index 000000000..2faaacdfd --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/review-sticker.php @@ -0,0 +1,19 @@ + + +
get_selector( 'review_sticker', $element ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ?>>
+ + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/rich-snippets-tpl.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/rich-snippets-tpl.php new file mode 100644 index 000000000..8457ab483 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/rich-snippets-tpl.php @@ -0,0 +1,22 @@ + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/rich-snippets.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/rich-snippets.php new file mode 100644 index 000000000..89cd06814 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/rich-snippets.php @@ -0,0 +1,18 @@ + + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/thankyou.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/thankyou.php new file mode 100644 index 000000000..2f7761b41 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/thankyou.php @@ -0,0 +1,57 @@ + + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/trustbadge-tpl.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/trustbadge-tpl.php new file mode 100644 index 000000000..71ed15bd1 --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/trustbadge-tpl.php @@ -0,0 +1,29 @@ + + + diff --git a/packages/woocommerce-trusted-shops/templates/trusted-shops/trustbadge.php b/packages/woocommerce-trusted-shops/templates/trusted-shops/trustbadge.php new file mode 100644 index 000000000..b5588808f --- /dev/null +++ b/packages/woocommerce-trusted-shops/templates/trusted-shops/trustbadge.php @@ -0,0 +1,17 @@ + + + diff --git a/packages/woocommerce-trusted-shops/woocommerce-trusted-shops.php b/packages/woocommerce-trusted-shops/woocommerce-trusted-shops.php new file mode 100644 index 000000000..941809722 --- /dev/null +++ b/packages/woocommerce-trusted-shops/woocommerce-trusted-shops.php @@ -0,0 +1,73 @@ + +
+

+ composer install', + '' . esc_html( str_replace( ABSPATH, '', __DIR__ ) ) . '' + ); + ?> +

+
+