Compare commits
	
		
			No commits in common. "2846bcbb1c788f3f19be980a525cff9b433fe28f" and "e20d0f05240ba6f6d1d1e7a8c49a7548948ad033" have entirely different histories.
		
	
	
		
			2846bcbb1c
			...
			e20d0f0524
		
	
		
| @ -272,75 +272,6 @@ def delete_brand(brand_id: int, db: Session = Depends(get_db)): | ||||
|     db.commit() | ||||
|     return {"message": "Brand deleted successfully"} | ||||
| 
 | ||||
| # BrandInShop endpoints | ||||
| @app.post("/brands-in-shops/", response_model=schemas.BrandInShop) | ||||
| def create_brand_in_shop(brand_in_shop: schemas.BrandInShopCreate, db: Session = Depends(get_db)): | ||||
|     # Validate shop exists | ||||
|     shop = db.query(models.Shop).filter(models.Shop.id == brand_in_shop.shop_id).first() | ||||
|     if shop is None: | ||||
|         raise HTTPException(status_code=404, detail="Shop not found") | ||||
|      | ||||
|     # Validate brand exists | ||||
|     brand = db.query(models.Brand).filter(models.Brand.id == brand_in_shop.brand_id).first() | ||||
|     if brand is None: | ||||
|         raise HTTPException(status_code=404, detail="Brand not found") | ||||
|      | ||||
|     # Check if this combination already exists | ||||
|     existing = db.query(models.BrandInShop).filter( | ||||
|         models.BrandInShop.shop_id == brand_in_shop.shop_id, | ||||
|         models.BrandInShop.brand_id == brand_in_shop.brand_id | ||||
|     ).first() | ||||
|     if existing: | ||||
|         raise HTTPException(status_code=400, detail="This brand is already associated with this shop") | ||||
|      | ||||
|     db_brand_in_shop = models.BrandInShop(**brand_in_shop.dict()) | ||||
|     db.add(db_brand_in_shop) | ||||
|     db.commit() | ||||
|     db.refresh(db_brand_in_shop) | ||||
|     return db_brand_in_shop | ||||
| 
 | ||||
| @app.get("/brands-in-shops/", response_model=List[schemas.BrandInShop]) | ||||
| def read_brands_in_shops(skip: int = 0, limit: int = 100, db: Session = Depends(get_db)): | ||||
|     brands_in_shops = db.query(models.BrandInShop).offset(skip).limit(limit).all() | ||||
|     return brands_in_shops | ||||
| 
 | ||||
| @app.get("/brands-in-shops/shop/{shop_id}", response_model=List[schemas.BrandInShop]) | ||||
| def read_brands_in_shop(shop_id: int, db: Session = Depends(get_db)): | ||||
|     # Validate shop exists | ||||
|     shop = db.query(models.Shop).filter(models.Shop.id == shop_id).first() | ||||
|     if shop is None: | ||||
|         raise HTTPException(status_code=404, detail="Shop not found") | ||||
|      | ||||
|     brands_in_shop = db.query(models.BrandInShop).filter(models.BrandInShop.shop_id == shop_id).all() | ||||
|     return brands_in_shop | ||||
| 
 | ||||
| @app.get("/brands-in-shops/brand/{brand_id}", response_model=List[schemas.BrandInShop]) | ||||
| def read_shops_with_brand(brand_id: int, db: Session = Depends(get_db)): | ||||
|     # Validate brand exists | ||||
|     brand = db.query(models.Brand).filter(models.Brand.id == brand_id).first() | ||||
|     if brand is None: | ||||
|         raise HTTPException(status_code=404, detail="Brand not found") | ||||
|      | ||||
|     shops_with_brand = db.query(models.BrandInShop).filter(models.BrandInShop.brand_id == brand_id).all() | ||||
|     return shops_with_brand | ||||
| 
 | ||||
| @app.get("/brands-in-shops/{brand_in_shop_id}", response_model=schemas.BrandInShop) | ||||
| def read_brand_in_shop(brand_in_shop_id: int, db: Session = Depends(get_db)): | ||||
|     brand_in_shop = db.query(models.BrandInShop).filter(models.BrandInShop.id == brand_in_shop_id).first() | ||||
|     if brand_in_shop is None: | ||||
|         raise HTTPException(status_code=404, detail="Brand in shop association not found") | ||||
|     return brand_in_shop | ||||
| 
 | ||||
| @app.delete("/brands-in-shops/{brand_in_shop_id}") | ||||
| def delete_brand_in_shop(brand_in_shop_id: int, db: Session = Depends(get_db)): | ||||
|     brand_in_shop = db.query(models.BrandInShop).filter(models.BrandInShop.id == brand_in_shop_id).first() | ||||
|     if brand_in_shop is None: | ||||
|         raise HTTPException(status_code=404, detail="Brand in shop association not found") | ||||
|      | ||||
|     db.delete(brand_in_shop) | ||||
|     db.commit() | ||||
|     return {"message": "Brand in shop association deleted successfully"} | ||||
| 
 | ||||
| # Grocery Category endpoints | ||||
| @app.post("/grocery-categories/", response_model=schemas.GroceryCategory) | ||||
| def create_grocery_category(category: schemas.GroceryCategoryCreate, db: Session = Depends(get_db)): | ||||
|  | ||||
| @ -17,19 +17,6 @@ shopping_event_products = Table( | ||||
|     Column('price', Float, nullable=False)  # Price of this product at the time of this shopping event | ||||
| ) | ||||
| 
 | ||||
| class BrandInShop(Base): | ||||
|     __tablename__ = "brands_in_shops" | ||||
|      | ||||
|     id = Column(Integer, primary_key=True, index=True) | ||||
|     shop_id = Column(Integer, ForeignKey("shops.id"), nullable=False) | ||||
|     brand_id = Column(Integer, ForeignKey("brands.id"), nullable=False) | ||||
|     created_at = Column(DateTime(timezone=True), server_default=func.now()) | ||||
|     updated_at = Column(DateTime(timezone=True), onupdate=func.now()) | ||||
|      | ||||
|     # Relationships | ||||
|     shop = relationship("Shop", back_populates="brands_in_shop") | ||||
|     brand = relationship("Brand", back_populates="shops_with_brand") | ||||
| 
 | ||||
| class Brand(Base): | ||||
|     __tablename__ = "brands" | ||||
|      | ||||
| @ -40,7 +27,6 @@ class Brand(Base): | ||||
|      | ||||
|     # Relationships | ||||
|     products = relationship("Product", back_populates="brand") | ||||
|     shops_with_brand = relationship("BrandInShop", back_populates="brand") | ||||
| 
 | ||||
| class GroceryCategory(Base): | ||||
|     __tablename__ = "grocery_categories" | ||||
| @ -96,7 +82,6 @@ class Shop(Base): | ||||
|      | ||||
|     # Relationships | ||||
|     shopping_events = relationship("ShoppingEvent", back_populates="shop") | ||||
|     brands_in_shop = relationship("BrandInShop", back_populates="shop") | ||||
| 
 | ||||
| class ShoppingEvent(Base): | ||||
|     __tablename__ = "shopping_events" | ||||
|  | ||||
| @ -20,28 +20,6 @@ class Brand(BrandBase): | ||||
|     class Config: | ||||
|         from_attributes = True | ||||
| 
 | ||||
| # BrandInShop schemas | ||||
| class BrandInShopBase(BaseModel): | ||||
|     shop_id: int | ||||
|     brand_id: int | ||||
| 
 | ||||
| class BrandInShopCreate(BrandInShopBase): | ||||
|     pass | ||||
| 
 | ||||
| class BrandInShopUpdate(BaseModel): | ||||
|     shop_id: Optional[int] = None | ||||
|     brand_id: Optional[int] = None | ||||
| 
 | ||||
| class BrandInShop(BrandInShopBase): | ||||
|     id: int | ||||
|     created_at: datetime | ||||
|     updated_at: Optional[datetime] = None | ||||
|     shop: "Shop" | ||||
|     brand: "Brand" | ||||
|      | ||||
|     class Config: | ||||
|         from_attributes = True | ||||
| 
 | ||||
| # Grocery Category schemas | ||||
| class GroceryCategoryBase(BaseModel): | ||||
|     name: str | ||||
| @ -190,7 +168,4 @@ class ShopStats(BaseModel): | ||||
|     shop_name: str | ||||
|     total_spent: float | ||||
|     visit_count: int | ||||
|     avg_per_visit: float | ||||
| 
 | ||||
| # Update forward references | ||||
| BrandInShop.model_rebuild()  | ||||
|     avg_per_visit: float  | ||||
| @ -1,6 +1,6 @@ | ||||
| <mxfile host="65bd71144e"> | ||||
|     <diagram name="Product Tracker Database Schema" id="database-schema"> | ||||
|         <mxGraphModel dx="1720" dy="739" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1169" pageHeight="827" math="0" shadow="0"> | ||||
|         <mxGraphModel dx="1848" dy="501" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1169" pageHeight="827" math="0" shadow="0"> | ||||
|             <root> | ||||
|                 <mxCell id="0"/> | ||||
|                 <mxCell id="1" parent="0"/> | ||||
| @ -58,7 +58,7 @@ | ||||
|                 <mxCell id="128" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" parent="2" vertex="1"> | ||||
|                     <mxGeometry y="90" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="129" value="FK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" parent="128" vertex="1"> | ||||
|                 <mxCell id="129" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" parent="128" vertex="1"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
| @ -71,7 +71,7 @@ | ||||
|                 <mxCell id="9" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" parent="2" vertex="1"> | ||||
|                     <mxGeometry y="120" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="10" value="FK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;whiteSpace=wrap;html=1;" parent="9" vertex="1"> | ||||
|                 <mxCell id="10" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;whiteSpace=wrap;html=1;" parent="9" vertex="1"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
| @ -390,7 +390,7 @@ | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="114" value="<span style="color: rgb(0, 0, 0); text-wrap-mode: wrap;">brands</span>" style="shape=table;startSize=30;container=1;collapsible=1;childLayout=tableLayout;fixedRows=1;rowLines=0;fontStyle=1;align=center;resizeLast=1;html=1;" parent="1" vertex="1"> | ||||
|                     <mxGeometry x="-430" y="414" width="180" height="150" as="geometry"/> | ||||
|                     <mxGeometry x="90" y="480" width="180" height="150" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="115" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=1;" parent="114" vertex="1"> | ||||
|                     <mxGeometry y="30" width="180" height="30" as="geometry"/> | ||||
| @ -483,7 +483,7 @@ | ||||
|                 <mxCell id="138" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" parent="131" vertex="1"> | ||||
|                     <mxGeometry y="90" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="139" value="FK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" parent="138" vertex="1"> | ||||
|                 <mxCell id="139" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" parent="138" vertex="1"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
| @ -526,150 +526,68 @@ | ||||
|                         <Array as="points"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="148" value="<span style="color: rgb(0, 0, 0); text-wrap-mode: wrap;">grocerie_categories</span>" style="shape=table;startSize=30;container=1;collapsible=1;childLayout=tableLayout;fixedRows=1;rowLines=0;fontStyle=1;align=center;resizeLast=1;html=1;" parent="1" vertex="1"> | ||||
|                 <mxCell id="148" value="<span style="color: rgb(0, 0, 0); text-wrap-mode: wrap;">grocerie_categories</span>" style="shape=table;startSize=30;container=1;collapsible=1;childLayout=tableLayout;fixedRows=1;rowLines=0;fontStyle=1;align=center;resizeLast=1;html=1;" vertex="1" parent="1"> | ||||
|                     <mxGeometry x="-210" y="715" width="180" height="150" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="149" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=1;" parent="148" vertex="1"> | ||||
|                 <mxCell id="149" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=1;" vertex="1" parent="148"> | ||||
|                     <mxGeometry y="30" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="150" value="PK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;fontStyle=1;overflow=hidden;whiteSpace=wrap;html=1;" parent="149" vertex="1"> | ||||
|                 <mxCell id="150" value="PK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;fontStyle=1;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="149"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="151" value="id: INTEGER" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;fontStyle=5;overflow=hidden;whiteSpace=wrap;html=1;" parent="149" vertex="1"> | ||||
|                 <mxCell id="151" value="id: INTEGER" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;fontStyle=5;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="149"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="152" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" parent="148" vertex="1"> | ||||
|                 <mxCell id="152" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" vertex="1" parent="148"> | ||||
|                     <mxGeometry y="60" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="153" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;whiteSpace=wrap;html=1;" parent="152" vertex="1"> | ||||
|                 <mxCell id="153" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="152"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="154" value="name: STRING" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;whiteSpace=wrap;html=1;" parent="152" vertex="1"> | ||||
|                 <mxCell id="154" value="name: STRING" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="152"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="155" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" parent="148" vertex="1"> | ||||
|                 <mxCell id="155" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" vertex="1" parent="148"> | ||||
|                     <mxGeometry y="90" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="156" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" parent="155" vertex="1"> | ||||
|                 <mxCell id="156" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" vertex="1" parent="155"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="157" value="created_at: DATETIME" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;" parent="155" vertex="1"> | ||||
|                 <mxCell id="157" value="created_at: DATETIME" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;" vertex="1" parent="155"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="158" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" parent="148" vertex="1"> | ||||
|                 <mxCell id="158" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" vertex="1" parent="148"> | ||||
|                     <mxGeometry y="120" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="159" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" parent="158" vertex="1"> | ||||
|                 <mxCell id="159" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" vertex="1" parent="158"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="160" value="updated_at: DATETIME" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;" parent="158" vertex="1"> | ||||
|                 <mxCell id="160" value="updated_at: DATETIME" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;" vertex="1" parent="158"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="161" value="" style="endArrow=ERmany;html=1;rounded=0;startArrow=ERone;startFill=0;endFill=0;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" parent="1" source="149" target="138" edge="1"> | ||||
|                 <mxCell id="161" value="" style="endArrow=ERmany;html=1;rounded=0;startArrow=ERone;startFill=0;endFill=0;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="149" target="138"> | ||||
|                     <mxGeometry width="50" height="50" relative="1" as="geometry"> | ||||
|                         <mxPoint x="270" y="785" as="sourcePoint"/> | ||||
|                         <mxPoint x="80" y="835" as="targetPoint"/> | ||||
|                         <Array as="points"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="199" value="" style="endArrow=ERmany;html=1;rounded=0;exitX=0;exitY=0.5;exitDx=0;exitDy=0;entryX=1;entryY=0.5;entryDx=0;entryDy=0;startArrow=ERone;startFill=0;endFill=0;" edge="1" parent="1" source="71" target="187"> | ||||
|                     <mxGeometry width="50" height="50" relative="1" as="geometry"> | ||||
|                         <mxPoint x="280" y="755" as="sourcePoint"/> | ||||
|                         <mxPoint x="430" y="615" as="targetPoint"/> | ||||
|                         <Array as="points"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="200" value="" style="endArrow=ERmany;html=1;rounded=0;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;startArrow=ERone;startFill=0;endFill=0;" edge="1" parent="1" source="115" target="190"> | ||||
|                     <mxGeometry width="50" height="50" relative="1" as="geometry"> | ||||
|                         <mxPoint x="90" y="135" as="sourcePoint"/> | ||||
|                         <mxPoint x="-21" y="352" as="targetPoint"/> | ||||
|                         <Array as="points"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="183" value="<span style="color: rgb(0, 0, 0); text-wrap-mode: wrap;">brands_in_shops</span>" style="shape=table;startSize=30;container=1;collapsible=1;childLayout=tableLayout;fixedRows=1;rowLines=0;fontStyle=1;align=center;resizeLast=1;html=1;" vertex="1" parent="1"> | ||||
|                     <mxGeometry x="-180" y="220" width="180" height="180" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="184" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=1;" vertex="1" parent="183"> | ||||
|                     <mxGeometry y="30" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="185" value="PK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;fontStyle=1;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="184"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="186" value="id: INTEGER" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;fontStyle=5;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="184"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="187" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" vertex="1" parent="183"> | ||||
|                     <mxGeometry y="60" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="188" value="FK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="187"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="189" value="<span style="color: rgb(0, 0, 0); text-wrap-mode: nowrap;">shop_id: INTEGER</span>" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;whiteSpace=wrap;html=1;" vertex="1" parent="187"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="190" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" vertex="1" parent="183"> | ||||
|                     <mxGeometry y="90" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="191" value="FK" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" vertex="1" parent="190"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="192" value="brand_id: INTEGER" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;" vertex="1" parent="190"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="193" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" vertex="1" parent="183"> | ||||
|                     <mxGeometry y="120" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="194" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" vertex="1" parent="193"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="195" value="created_at: DATETIME" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;" vertex="1" parent="193"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="196" value="" style="shape=tableRow;horizontal=0;startSize=0;swimlaneHead=0;swimlaneBody=0;fillColor=none;collapsible=0;dropTarget=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;top=0;left=0;right=0;bottom=0;" vertex="1" parent="183"> | ||||
|                     <mxGeometry y="150" width="180" height="30" as="geometry"/> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="197" value="" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;editable=1;overflow=hidden;" vertex="1" parent="196"> | ||||
|                     <mxGeometry width="30" height="30" as="geometry"> | ||||
|                         <mxRectangle width="30" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|                 <mxCell id="198" value="updated_at: DATETIME" style="shape=partialRectangle;connectable=0;fillColor=none;top=0;left=0;bottom=0;right=0;align=left;spacingLeft=6;overflow=hidden;" vertex="1" parent="196"> | ||||
|                     <mxGeometry x="30" width="150" height="30" as="geometry"> | ||||
|                         <mxRectangle width="150" height="30" as="alternateBounds"/> | ||||
|                     </mxGeometry> | ||||
|                 </mxCell> | ||||
|             </root> | ||||
|         </mxGraphModel> | ||||
|     </diagram> | ||||
|  | ||||
| @ -4,6 +4,7 @@ import Dashboard from './components/Dashboard'; | ||||
| import ShopList from './components/ShopList'; | ||||
| import ProductList from './components/ProductList'; | ||||
| import ShoppingEventList from './components/ShoppingEventList'; | ||||
| import ShoppingEventForm from './components/ShoppingEventForm'; | ||||
| import BrandList from './components/BrandList'; | ||||
| import GroceryList from './components/GroceryList'; | ||||
| import GroceryCategoryList from './components/GroceryCategoryList'; | ||||
| @ -107,6 +108,8 @@ function App() { | ||||
|             <Routes> | ||||
|               <Route path="/" element={<Dashboard />} /> | ||||
|               <Route path="/shopping-events" element={<ShoppingEventList />} /> | ||||
|               <Route path="/shopping-events/new" element={<ShoppingEventForm />} /> | ||||
|               <Route path="/shopping-events/:id/edit" element={<ShoppingEventForm />} /> | ||||
|               <Route path="/shops" element={<ShopList />} /> | ||||
|               <Route path="/products" element={<ProductList />} /> | ||||
|               <Route path="/brands" element={<BrandList />} /> | ||||
|  | ||||
| @ -36,26 +36,6 @@ const AddBrandModal: React.FC<AddBrandModalProps> = ({ isOpen, onClose, onBrandA | ||||
|     setError(''); | ||||
|   }, [editBrand, isOpen]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (!isOpen) return; | ||||
| 
 | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (event.key === 'Escape') { | ||||
|         onClose(); | ||||
|       } else if (event.key === 'Enter' && !event.shiftKey && !loading) { | ||||
|         event.preventDefault(); | ||||
|         if (formData.name.trim()) { | ||||
|           handleSubmit(event as any); | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     document.addEventListener('keydown', handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     }; | ||||
|   }, [isOpen, formData, loading, onClose]); | ||||
| 
 | ||||
|   const handleSubmit = async (e: React.FormEvent) => { | ||||
|     e.preventDefault(); | ||||
|     if (!formData.name.trim()) { | ||||
|  | ||||
| @ -24,24 +24,6 @@ const AddGroceryCategoryModal: React.FC<AddGroceryCategoryModalProps> = ({ categ | ||||
|     } | ||||
|   }, [category]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (event.key === 'Escape') { | ||||
|         onClose(); | ||||
|       } else if (event.key === 'Enter' && !event.shiftKey && !loading) { | ||||
|         event.preventDefault(); | ||||
|         if (formData.name.trim()) { | ||||
|           handleSubmit(event as any); | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     document.addEventListener('keydown', handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     }; | ||||
|   }, [formData, loading, onClose]); | ||||
| 
 | ||||
|   const handleSubmit = async (e: React.FormEvent) => { | ||||
|     e.preventDefault(); | ||||
|     setLoading(true); | ||||
| @ -62,7 +44,6 @@ const AddGroceryCategoryModal: React.FC<AddGroceryCategoryModalProps> = ({ categ | ||||
|     } catch (error) { | ||||
|       console.error('Error saving category:', error); | ||||
|       setMessage(`Error ${isEditMode ? 'updating' : 'creating'} category. Please try again.`); | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } finally { | ||||
|       setLoading(false); | ||||
|     } | ||||
| @ -77,10 +58,10 @@ const AddGroceryCategoryModal: React.FC<AddGroceryCategoryModalProps> = ({ categ | ||||
|           </h3> | ||||
| 
 | ||||
|           {message && ( | ||||
|             <div className={`mb-4 px-4 py-3 rounded ${ | ||||
|             <div className={`mb-4 p-4 rounded-md ${ | ||||
|               message.includes('Error')  | ||||
|                 ? 'bg-red-50 border border-red-200 text-red-700'  | ||||
|                 : 'bg-green-50 border border-green-200 text-green-700' | ||||
|                 ? 'bg-red-50 text-red-700'  | ||||
|                 : 'bg-green-50 text-green-700' | ||||
|             }`}>
 | ||||
|               {message} | ||||
|             </div> | ||||
|  | ||||
| @ -28,24 +28,6 @@ const AddGroceryModal: React.FC<AddGroceryModalProps> = ({ grocery, onClose }) = | ||||
|     } | ||||
|   }, [grocery]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (event.key === 'Escape') { | ||||
|         onClose(); | ||||
|       } else if (event.key === 'Enter' && !event.shiftKey && !loading) { | ||||
|         event.preventDefault(); | ||||
|         if (formData.name.trim() && formData.category_id > 0) { | ||||
|           handleSubmit(event as any); | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     document.addEventListener('keydown', handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     }; | ||||
|   }, [formData, loading, onClose]); | ||||
| 
 | ||||
|   const fetchCategories = async () => { | ||||
|     try { | ||||
|       const response = await groceryCategoryApi.getAll(); | ||||
| @ -53,7 +35,6 @@ const AddGroceryModal: React.FC<AddGroceryModalProps> = ({ grocery, onClose }) = | ||||
|     } catch (error) { | ||||
|       console.error('Error fetching categories:', error); | ||||
|       setMessage('Error loading categories. Please try again.'); | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
| @ -77,7 +58,6 @@ const AddGroceryModal: React.FC<AddGroceryModalProps> = ({ grocery, onClose }) = | ||||
|     } catch (error) { | ||||
|       console.error('Error saving grocery:', error); | ||||
|       setMessage(`Error ${isEditMode ? 'updating' : 'creating'} grocery. Please try again.`); | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } finally { | ||||
|       setLoading(false); | ||||
|     } | ||||
| @ -92,10 +72,10 @@ const AddGroceryModal: React.FC<AddGroceryModalProps> = ({ grocery, onClose }) = | ||||
|           </h3> | ||||
| 
 | ||||
|           {message && ( | ||||
|             <div className={`mb-4 px-4 py-3 rounded ${ | ||||
|             <div className={`mb-4 p-4 rounded-md ${ | ||||
|               message.includes('Error')  | ||||
|                 ? 'bg-red-50 border border-red-200 text-red-700'  | ||||
|                 : 'bg-green-50 border border-green-200 text-green-700' | ||||
|                 ? 'bg-red-50 text-red-700'  | ||||
|                 : 'bg-green-50 text-green-700' | ||||
|             }`}>
 | ||||
|               {message} | ||||
|             </div> | ||||
|  | ||||
| @ -85,26 +85,6 @@ const AddProductModal: React.FC<AddProductModalProps> = ({ isOpen, onClose, onPr | ||||
|     setError(''); | ||||
|   }, [editProduct, isOpen]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (!isOpen) return; | ||||
| 
 | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (event.key === 'Escape') { | ||||
|         onClose(); | ||||
|       } else if (event.key === 'Enter' && !event.shiftKey && !loading) { | ||||
|         event.preventDefault(); | ||||
|         if (formData.name.trim() && formData.grocery_id) { | ||||
|           handleSubmit(event as any); | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     document.addEventListener('keydown', handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     }; | ||||
|   }, [isOpen, formData, loading, onClose]); | ||||
| 
 | ||||
|   const handleSubmit = async (e: React.FormEvent) => { | ||||
|     e.preventDefault(); | ||||
|     if (!formData.name.trim() || !formData.grocery_id) { | ||||
|  | ||||
| @ -1,6 +1,6 @@ | ||||
| import React, { useState, useEffect } from 'react'; | ||||
| import { shopApi, brandApi, brandInShopApi } from '../services/api'; | ||||
| import { Shop, Brand, BrandInShop } from '../types'; | ||||
| import { shopApi } from '../services/api'; | ||||
| import { Shop } from '../types'; | ||||
| 
 | ||||
| interface AddShopModalProps { | ||||
|   isOpen: boolean; | ||||
| @ -13,99 +13,37 @@ interface ShopFormData { | ||||
|   name: string; | ||||
|   city: string; | ||||
|   address?: string; | ||||
|   selectedBrands: number[]; | ||||
| } | ||||
| 
 | ||||
| const AddShopModal: React.FC<AddShopModalProps> = ({ isOpen, onClose, onShopAdded, editShop }) => { | ||||
|   const [formData, setFormData] = useState<ShopFormData>({ | ||||
|     name: '', | ||||
|     city: '', | ||||
|     address: '', | ||||
|     selectedBrands: [] | ||||
|     address: '' | ||||
|   }); | ||||
|   const [brands, setBrands] = useState<Brand[]>([]); | ||||
|   const [loading, setLoading] = useState(false); | ||||
|   const [error, setError] = useState(''); | ||||
| 
 | ||||
|   const isEditMode = !!editShop; | ||||
| 
 | ||||
|   // Load brands when modal opens
 | ||||
|   useEffect(() => { | ||||
|     if (isOpen) { | ||||
|       fetchBrands(); | ||||
|       if (editShop) { | ||||
|         loadShopBrands(editShop.id); | ||||
|       } | ||||
|     } | ||||
|   }, [isOpen, editShop]); | ||||
| 
 | ||||
|   const fetchBrands = async () => { | ||||
|     try { | ||||
|       const response = await brandApi.getAll(); | ||||
|       setBrands(response.data); | ||||
|     } catch (err) { | ||||
|       console.error('Error fetching brands:', err); | ||||
|       setError('Failed to load brands. Please try again.'); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const loadShopBrands = async (shopId: number) => { | ||||
|     try { | ||||
|       const response = await brandInShopApi.getByShop(shopId); | ||||
|       const brandIds = response.data.map((brandInShop: BrandInShop) => brandInShop.brand_id); | ||||
|       setFormData(prev => ({ | ||||
|         ...prev, | ||||
|         selectedBrands: brandIds | ||||
|       })); | ||||
|     } catch (err) { | ||||
|       console.error('Error loading shop brands:', err); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   // Initialize form data when editing
 | ||||
|   useEffect(() => { | ||||
|     if (editShop) { | ||||
|       setFormData(prev => ({ | ||||
|         ...prev, | ||||
|       setFormData({ | ||||
|         name: editShop.name, | ||||
|         city: editShop.city, | ||||
|         address: editShop.address || '' | ||||
|       })); | ||||
|       }); | ||||
|     } else { | ||||
|       setFormData({ | ||||
|         name: '', | ||||
|         city: '', | ||||
|         address: '', | ||||
|         selectedBrands: [] | ||||
|         address: '' | ||||
|       }); | ||||
|     } | ||||
|     setError(''); | ||||
|   }, [editShop, isOpen]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (!isOpen) return; | ||||
| 
 | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (event.key === 'Escape') { | ||||
|         onClose(); | ||||
|       } else if (event.key === 'Enter' && !event.shiftKey && !loading) { | ||||
|         // Only trigger submit if not in a textarea and form is valid
 | ||||
|         const target = event.target as HTMLElement; | ||||
|         if (target.tagName !== 'TEXTAREA') { | ||||
|           event.preventDefault(); | ||||
|           if (formData.name.trim() && formData.city.trim()) { | ||||
|             handleSubmit(event as any); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     document.addEventListener('keydown', handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     }; | ||||
|   }, [isOpen, formData, loading, onClose]); | ||||
| 
 | ||||
|   const handleSubmit = async (e: React.FormEvent) => { | ||||
|     e.preventDefault(); | ||||
|     if (!formData.name.trim() || !formData.city.trim()) { | ||||
| @ -124,48 +62,17 @@ const AddShopModal: React.FC<AddShopModalProps> = ({ isOpen, onClose, onShopAdde | ||||
|         address: trimmedAddress && trimmedAddress.length > 0 ? trimmedAddress : null | ||||
|       }; | ||||
|        | ||||
|       let shopId: number; | ||||
|        | ||||
|       if (isEditMode && editShop) { | ||||
|         const updatedShop = await shopApi.update(editShop.id, shopData); | ||||
|         shopId = editShop.id; | ||||
|         await shopApi.update(editShop.id, shopData); | ||||
|       } else { | ||||
|         const newShop = await shopApi.create(shopData); | ||||
|         shopId = newShop.data.id; | ||||
|       } | ||||
| 
 | ||||
|       // Handle brand associations
 | ||||
|       if (isEditMode && editShop) { | ||||
|         // Get existing brand associations
 | ||||
|         const existingBrands = await brandInShopApi.getByShop(editShop.id); | ||||
|         const existingBrandIds = existingBrands.data.map(b => b.brand_id); | ||||
|          | ||||
|         // Remove brands that are no longer selected
 | ||||
|         for (const brandInShop of existingBrands.data) { | ||||
|           if (!formData.selectedBrands.includes(brandInShop.brand_id)) { | ||||
|             await brandInShopApi.delete(brandInShop.id); | ||||
|           } | ||||
|         } | ||||
|          | ||||
|         // Add new brand associations
 | ||||
|         for (const brandId of formData.selectedBrands) { | ||||
|           if (!existingBrandIds.includes(brandId)) { | ||||
|             await brandInShopApi.create({ shop_id: shopId, brand_id: brandId }); | ||||
|           } | ||||
|         } | ||||
|       } else { | ||||
|         // Create new brand associations for new shop
 | ||||
|         for (const brandId of formData.selectedBrands) { | ||||
|           await brandInShopApi.create({ shop_id: shopId, brand_id: brandId }); | ||||
|         } | ||||
|         await shopApi.create(shopData); | ||||
|       } | ||||
|        | ||||
|       // Reset form
 | ||||
|       setFormData({ | ||||
|         name: '', | ||||
|         city: '', | ||||
|         address: '', | ||||
|         selectedBrands: [] | ||||
|         address: '' | ||||
|       }); | ||||
|        | ||||
|       onShopAdded(); | ||||
| @ -186,20 +93,11 @@ const AddShopModal: React.FC<AddShopModalProps> = ({ isOpen, onClose, onShopAdde | ||||
|     })); | ||||
|   }; | ||||
| 
 | ||||
|   const handleBrandToggle = (brandId: number) => { | ||||
|     setFormData(prev => ({ | ||||
|       ...prev, | ||||
|       selectedBrands: prev.selectedBrands.includes(brandId) | ||||
|         ? prev.selectedBrands.filter(id => id !== brandId) | ||||
|         : [...prev.selectedBrands, brandId] | ||||
|     })); | ||||
|   }; | ||||
| 
 | ||||
|   if (!isOpen) return null; | ||||
| 
 | ||||
|   return ( | ||||
|     <div className="fixed inset-0 bg-gray-600 bg-opacity-50 overflow-y-auto h-full w-full z-50"> | ||||
|       <div className="relative top-20 mx-auto p-5 border w-96 shadow-lg rounded-md bg-white max-h-[80vh] overflow-y-auto"> | ||||
|       <div className="relative top-20 mx-auto p-5 border w-96 shadow-lg rounded-md bg-white"> | ||||
|         <div className="mt-3"> | ||||
|           <div className="flex justify-between items-center mb-4"> | ||||
|             <h3 className="text-lg font-medium text-gray-900"> | ||||
| @ -269,34 +167,6 @@ const AddShopModal: React.FC<AddShopModalProps> = ({ isOpen, onClose, onShopAdde | ||||
|               /> | ||||
|             </div> | ||||
| 
 | ||||
|             <div> | ||||
|               <label className="block text-sm font-medium text-gray-700 mb-2"> | ||||
|                 Available Brands (Optional) | ||||
|               </label> | ||||
|               <div className="max-h-40 overflow-y-auto border border-gray-300 rounded-md p-3 bg-gray-50"> | ||||
|                 {brands.length === 0 ? ( | ||||
|                   <p className="text-sm text-gray-500">Loading brands...</p> | ||||
|                 ) : ( | ||||
|                   <div className="space-y-2"> | ||||
|                     {brands.map(brand => ( | ||||
|                       <label key={brand.id} className="flex items-center"> | ||||
|                         <input | ||||
|                           type="checkbox" | ||||
|                           checked={formData.selectedBrands.includes(brand.id)} | ||||
|                           onChange={() => handleBrandToggle(brand.id)} | ||||
|                           className="h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300 rounded" | ||||
|                         /> | ||||
|                         <span className="ml-2 text-sm text-gray-900">{brand.name}</span> | ||||
|                       </label> | ||||
|                     ))} | ||||
|                   </div> | ||||
|                 )} | ||||
|               </div> | ||||
|               <p className="mt-1 text-xs text-gray-500"> | ||||
|                 Select the brands that are available in this shop | ||||
|               </p> | ||||
|             </div> | ||||
| 
 | ||||
|             <div className="flex justify-end space-x-3 pt-4"> | ||||
|               <button | ||||
|                 type="button" | ||||
|  | ||||
| @ -14,8 +14,6 @@ const BrandList: React.FC = () => { | ||||
|   const [editingBrand, setEditingBrand] = useState<Brand | null>(null); | ||||
|   const [deletingBrand, setDeletingBrand] = useState<Brand | null>(null); | ||||
|   const [deleteLoading, setDeleteLoading] = useState(false); | ||||
|   const [sortField, setSortField] = useState<keyof Brand>('name'); | ||||
|   const [sortDirection, setSortDirection] = useState<'asc' | 'desc'>('asc'); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchBrands(); | ||||
| @ -84,58 +82,6 @@ const BrandList: React.FC = () => { | ||||
|     setDeletingBrand(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleSort = (field: keyof Brand) => { | ||||
|     if (field === sortField) { | ||||
|       setSortDirection(sortDirection === 'asc' ? 'desc' : 'asc'); | ||||
|     } else { | ||||
|       setSortField(field); | ||||
|       setSortDirection('asc'); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const sortedBrands = [...brands].sort((a, b) => { | ||||
|     let aValue = a[sortField]; | ||||
|     let bValue = b[sortField]; | ||||
| 
 | ||||
|     // Handle null/undefined values
 | ||||
|     if (aValue === null || aValue === undefined) aValue = ''; | ||||
|     if (bValue === null || bValue === undefined) bValue = ''; | ||||
| 
 | ||||
|     // Convert to string for comparison
 | ||||
|     const aStr = String(aValue).toLowerCase(); | ||||
|     const bStr = String(bValue).toLowerCase(); | ||||
| 
 | ||||
|     if (sortDirection === 'asc') { | ||||
|       return aStr.localeCompare(bStr); | ||||
|     } else { | ||||
|       return bStr.localeCompare(aStr); | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const getSortIcon = (field: keyof Brand) => { | ||||
|     if (sortField !== field) { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 16V4m0 0L3 8m4-4l4 4m6 0v12m0 0l4-4m-4 4l-4-4" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|      | ||||
|     if (sortDirection === 'asc') { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 15l7-7 7 7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } else { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   if (loading) { | ||||
|     return ( | ||||
|       <div className="flex justify-center items-center h-64"> | ||||
| @ -172,73 +118,47 @@ const BrandList: React.FC = () => { | ||||
|             <p className="mt-1 text-sm text-gray-500">Get started by adding your first brand.</p> | ||||
|           </div> | ||||
|         ) : ( | ||||
|           <table className="min-w-full divide-y divide-gray-200"> | ||||
|             <thead className="bg-gray-50"> | ||||
|               <tr> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('name')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Name | ||||
|                     {getSortIcon('name')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('created_at')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Created | ||||
|                     {getSortIcon('created_at')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('updated_at')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Updated | ||||
|                     {getSortIcon('updated_at')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Actions | ||||
|                 </th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody className="bg-white divide-y divide-gray-200"> | ||||
|               {sortedBrands.map((brand) => ( | ||||
|                 <tr key={brand.id} className="hover:bg-gray-50"> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <div className="text-sm font-medium text-gray-900"> | ||||
|                       {brand.name} | ||||
|                     </div> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                     {new Date(brand.created_at).toLocaleDateString()} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                     {brand.updated_at ? new Date(brand.updated_at).toLocaleDateString() : '-'} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm font-medium"> | ||||
|           <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6 p-6"> | ||||
|             {brands.map((brand) => ( | ||||
|               <div key={brand.id} className="bg-white border border-gray-200 rounded-lg p-6 hover:shadow-md transition-shadow"> | ||||
|                 <div className="flex items-center justify-between mb-4"> | ||||
|                   <h3 className="text-lg font-medium text-gray-900">{brand.name}</h3> | ||||
|                   <div className="flex space-x-2"> | ||||
|                     <button  | ||||
|                       onClick={() => handleEditBrand(brand)} | ||||
|                       className="text-indigo-600 hover:text-indigo-900 mr-3" | ||||
|                       className="text-indigo-600 hover:text-indigo-900 text-sm" | ||||
|                     > | ||||
|                       Edit | ||||
|                     </button> | ||||
|                     <button  | ||||
|                       onClick={() => handleDeleteBrand(brand)} | ||||
|                       className="text-red-600 hover:text-red-900" | ||||
|                       className="text-red-600 hover:text-red-900 text-sm" | ||||
|                     > | ||||
|                       Delete | ||||
|                     </button> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|             </tbody> | ||||
|           </table> | ||||
|                   </div> | ||||
|                 </div> | ||||
|                  | ||||
|                 <div className="space-y-2"> | ||||
|                   <div className="flex items-center text-sm text-gray-600"> | ||||
|                     <svg className="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                       <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z" /> | ||||
|                     </svg> | ||||
|                     Added {new Date(brand.created_at).toLocaleDateString()} | ||||
|                   </div> | ||||
|                    | ||||
|                   {brand.updated_at && ( | ||||
|                     <div className="flex items-center text-sm text-gray-600"> | ||||
|                       <svg className="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                         <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15" /> | ||||
|                       </svg> | ||||
|                       Updated {new Date(brand.updated_at).toLocaleDateString()} | ||||
|                     </div> | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|             ))} | ||||
|           </div> | ||||
|         )} | ||||
|       </div> | ||||
| 
 | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import React, { useEffect } from 'react'; | ||||
| import React from 'react'; | ||||
| 
 | ||||
| interface ConfirmDeleteModalProps { | ||||
|   isOpen: boolean; | ||||
| @ -17,24 +17,6 @@ const ConfirmDeleteModal: React.FC<ConfirmDeleteModalProps> = ({ | ||||
|   message, | ||||
|   isLoading = false | ||||
| }) => { | ||||
|   useEffect(() => { | ||||
|     if (!isOpen) return; | ||||
| 
 | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (event.key === 'Escape') { | ||||
|         onClose(); | ||||
|       } else if (event.key === 'Enter' && !isLoading) { | ||||
|         event.preventDefault(); | ||||
|         onConfirm(); | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     document.addEventListener('keydown', handleKeyDown); | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     }; | ||||
|   }, [isOpen, isLoading, onClose, onConfirm]); | ||||
| 
 | ||||
|   if (!isOpen) return null; | ||||
| 
 | ||||
|   return ( | ||||
|  | ||||
| @ -102,7 +102,7 @@ const Dashboard: React.FC = () => { | ||||
|         <div className="p-6"> | ||||
|           <div className="grid grid-cols-1 md:grid-cols-3 gap-4"> | ||||
|             <button  | ||||
|               onClick={() => navigate('/shopping-events?add=true')} | ||||
|               onClick={() => navigate('/shopping-events/new')} | ||||
|               className="flex items-center p-4 border border-gray-300 rounded-lg hover:bg-gray-50 transition-colors" | ||||
|             > | ||||
|               <div className="p-2 bg-blue-100 rounded-md mr-3"> | ||||
|  | ||||
| @ -2,7 +2,6 @@ import React, { useState, useEffect } from 'react'; | ||||
| import { GroceryCategory } from '../types'; | ||||
| import { groceryCategoryApi } from '../services/api'; | ||||
| import AddGroceryCategoryModal from './AddGroceryCategoryModal'; | ||||
| import ConfirmDeleteModal from './ConfirmDeleteModal'; | ||||
| 
 | ||||
| const GroceryCategoryList: React.FC = () => { | ||||
|   const [categories, setCategories] = useState<GroceryCategory[]>([]); | ||||
| @ -10,10 +9,6 @@ const GroceryCategoryList: React.FC = () => { | ||||
|   const [message, setMessage] = useState(''); | ||||
|   const [isModalOpen, setIsModalOpen] = useState(false); | ||||
|   const [editingCategory, setEditingCategory] = useState<GroceryCategory | null>(null); | ||||
|   const [deletingCategory, setDeletingCategory] = useState<GroceryCategory | null>(null); | ||||
|   const [deleteLoading, setDeleteLoading] = useState(false); | ||||
|   const [sortField, setSortField] = useState<keyof GroceryCategory>('name'); | ||||
|   const [sortDirection, setSortDirection] = useState<'asc' | 'desc'>('asc'); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchCategories(); | ||||
| @ -32,37 +27,25 @@ const GroceryCategoryList: React.FC = () => { | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const handleDelete = async (category: GroceryCategory) => { | ||||
|     setDeletingCategory(category); | ||||
|   }; | ||||
| 
 | ||||
|   const confirmDelete = async () => { | ||||
|     if (!deletingCategory) return; | ||||
| 
 | ||||
|     try { | ||||
|       setDeleteLoading(true); | ||||
|       await groceryCategoryApi.delete(deletingCategory.id); | ||||
|       setMessage('Category deleted successfully!'); | ||||
|       setDeletingCategory(null); | ||||
|       fetchCategories(); | ||||
|       setTimeout(() => setMessage(''), 1500); | ||||
|     } catch (error: any) { | ||||
|       console.error('Error deleting category:', error); | ||||
|       if (error.response?.status === 400) { | ||||
|         setMessage('Cannot delete category: groceries are still associated with this category.'); | ||||
|       } else { | ||||
|         setMessage('Error deleting category. Please try again.'); | ||||
|   const handleDelete = async (id: number) => { | ||||
|     if (window.confirm('Are you sure you want to delete this category?')) { | ||||
|       try { | ||||
|         await groceryCategoryApi.delete(id); | ||||
|         setMessage('Category deleted successfully!'); | ||||
|         fetchCategories(); | ||||
|         setTimeout(() => setMessage(''), 3000); | ||||
|       } catch (error: any) { | ||||
|         console.error('Error deleting category:', error); | ||||
|         if (error.response?.status === 400) { | ||||
|           setMessage('Cannot delete category: groceries are still associated with this category.'); | ||||
|         } else { | ||||
|           setMessage('Error deleting category. Please try again.'); | ||||
|         } | ||||
|         setTimeout(() => setMessage(''), 5000); | ||||
|       } | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } finally { | ||||
|       setDeleteLoading(false); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const handleCloseDeleteModal = () => { | ||||
|     setDeletingCategory(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleEdit = (category: GroceryCategory) => { | ||||
|     setEditingCategory(category); | ||||
|     setIsModalOpen(true); | ||||
| @ -74,58 +57,6 @@ const GroceryCategoryList: React.FC = () => { | ||||
|     fetchCategories(); | ||||
|   }; | ||||
| 
 | ||||
|   const handleSort = (field: keyof GroceryCategory) => { | ||||
|     if (field === sortField) { | ||||
|       setSortDirection(sortDirection === 'asc' ? 'desc' : 'asc'); | ||||
|     } else { | ||||
|       setSortField(field); | ||||
|       setSortDirection('asc'); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const sortedCategories = [...categories].sort((a, b) => { | ||||
|     let aValue = a[sortField]; | ||||
|     let bValue = b[sortField]; | ||||
| 
 | ||||
|     // Handle null/undefined values
 | ||||
|     if (aValue === null || aValue === undefined) aValue = ''; | ||||
|     if (bValue === null || bValue === undefined) bValue = ''; | ||||
| 
 | ||||
|     // Convert to string for comparison
 | ||||
|     const aStr = String(aValue).toLowerCase(); | ||||
|     const bStr = String(bValue).toLowerCase(); | ||||
| 
 | ||||
|     if (sortDirection === 'asc') { | ||||
|       return aStr.localeCompare(bStr); | ||||
|     } else { | ||||
|       return bStr.localeCompare(aStr); | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const getSortIcon = (field: keyof GroceryCategory) => { | ||||
|     if (sortField !== field) { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 16V4m0 0L3 8m4-4l4 4m6 0v12m0 0l4-4m-4 4l-4-4" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|      | ||||
|     if (sortDirection === 'asc') { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 15l7-7 7 7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } else { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   if (loading) { | ||||
|     return ( | ||||
|       <div className="flex justify-center items-center h-64"> | ||||
| @ -135,93 +66,81 @@ const GroceryCategoryList: React.FC = () => { | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <div className="space-y-6"> | ||||
|       <div className="flex justify-between items-center"> | ||||
|         <h1 className="text-2xl font-bold text-gray-900">Grocery Categories</h1> | ||||
|         <button | ||||
|           onClick={() => setIsModalOpen(true)} | ||||
|           className="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded" | ||||
|         > | ||||
|           Add New Category | ||||
|         </button> | ||||
|       </div> | ||||
| 
 | ||||
|       {message && ( | ||||
|         <div className={`px-4 py-3 rounded ${ | ||||
|           message.includes('Error') || message.includes('Cannot')  | ||||
|             ? 'bg-red-50 border border-red-200 text-red-700'  | ||||
|             : 'bg-green-50 border border-green-200 text-green-700' | ||||
|         }`}>
 | ||||
|           {message} | ||||
|         </div> | ||||
|       )} | ||||
| 
 | ||||
|       <div className="bg-white shadow rounded-lg overflow-hidden"> | ||||
|         {categories.length === 0 ? ( | ||||
|           <div className="text-center py-12"> | ||||
|             <svg className="mx-auto h-12 w-12 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 48 48"> | ||||
|               <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 7h.01M7 3h5c.512 0 1.024.195 1.414.586l7 7a2 2 0 010 2.828l-7 7a1.994 1.994 0 01-1.414.586H7m0-18v18m0-18h.01" /> | ||||
|             </svg> | ||||
|             <h3 className="mt-2 text-sm font-medium text-gray-900">No categories</h3> | ||||
|             <p className="mt-1 text-sm text-gray-500">Get started by adding your first category.</p> | ||||
|     <div className="max-w-4xl mx-auto"> | ||||
|       <div className="bg-white shadow rounded-lg"> | ||||
|         <div className="px-4 py-5 sm:p-6"> | ||||
|           <div className="flex justify-between items-center mb-4"> | ||||
|             <h3 className="text-lg leading-6 font-medium text-gray-900"> | ||||
|               Grocery Categories | ||||
|             </h3> | ||||
|             <button | ||||
|               onClick={() => setIsModalOpen(true)} | ||||
|               className="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded" | ||||
|             > | ||||
|               Add Category | ||||
|             </button> | ||||
|           </div> | ||||
|         ) : ( | ||||
|           <table className="min-w-full divide-y divide-gray-200"> | ||||
|             <thead className="bg-gray-50"> | ||||
|               <tr> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('name')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Name | ||||
|                     {getSortIcon('name')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('created_at')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Created | ||||
|                     {getSortIcon('created_at')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Actions | ||||
|                 </th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody className="bg-white divide-y divide-gray-200"> | ||||
|               {sortedCategories.map((category) => ( | ||||
|                 <tr key={category.id} className="hover:bg-gray-50"> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <div className="text-sm font-medium text-gray-900"> | ||||
|                       {category.name} | ||||
|                     </div> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                     {new Date(category.created_at).toLocaleDateString()} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm font-medium"> | ||||
|                     <button | ||||
|                       onClick={() => handleEdit(category)} | ||||
|                       className="text-indigo-600 hover:text-indigo-900 mr-3" | ||||
|                     > | ||||
|                       Edit | ||||
|                     </button> | ||||
|                     <button | ||||
|                       onClick={() => handleDelete(category)} | ||||
|                       className="text-red-600 hover:text-red-900" | ||||
|                     > | ||||
|                       Delete | ||||
|                     </button> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|             </tbody> | ||||
|           </table> | ||||
|         )} | ||||
| 
 | ||||
|           {message && ( | ||||
|             <div className={`mb-4 p-4 rounded-md ${ | ||||
|               message.includes('Error') || message.includes('Cannot')  | ||||
|                 ? 'bg-red-50 text-red-700'  | ||||
|                 : 'bg-green-50 text-green-700' | ||||
|             }`}>
 | ||||
|               {message} | ||||
|             </div> | ||||
|           )} | ||||
| 
 | ||||
|           {categories.length === 0 ? ( | ||||
|             <div className="text-center py-8"> | ||||
|               <p className="text-gray-500">No categories found. Add your first category!</p> | ||||
|             </div> | ||||
|           ) : ( | ||||
|             <div className="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg"> | ||||
|               <table className="min-w-full divide-y divide-gray-300"> | ||||
|                 <thead className="bg-gray-50"> | ||||
|                   <tr> | ||||
|                     <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                       Name | ||||
|                     </th> | ||||
|                     <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                       Created | ||||
|                     </th> | ||||
|                     <th className="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                       Actions | ||||
|                     </th> | ||||
|                   </tr> | ||||
|                 </thead> | ||||
|                 <tbody className="bg-white divide-y divide-gray-200"> | ||||
|                   {categories.map((category) => ( | ||||
|                     <tr key={category.id} className="hover:bg-gray-50"> | ||||
|                       <td className="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900"> | ||||
|                         {category.name} | ||||
|                       </td> | ||||
|                       <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                         {new Date(category.created_at).toLocaleDateString()} | ||||
|                       </td> | ||||
|                       <td className="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||
|                         <button | ||||
|                           onClick={() => handleEdit(category)} | ||||
|                           className="text-indigo-600 hover:text-indigo-900 mr-4" | ||||
|                         > | ||||
|                           Edit | ||||
|                         </button> | ||||
|                         <button | ||||
|                           onClick={() => handleDelete(category.id)} | ||||
|                           className="text-red-600 hover:text-red-900" | ||||
|                         > | ||||
|                           Delete | ||||
|                         </button> | ||||
|                       </td> | ||||
|                     </tr> | ||||
|                   ))} | ||||
|                 </tbody> | ||||
|               </table> | ||||
|             </div> | ||||
|           )} | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       {isModalOpen && ( | ||||
| @ -230,15 +149,6 @@ const GroceryCategoryList: React.FC = () => { | ||||
|           onClose={handleModalClose} | ||||
|         /> | ||||
|       )} | ||||
| 
 | ||||
|       <ConfirmDeleteModal | ||||
|         isOpen={!!deletingCategory} | ||||
|         onClose={handleCloseDeleteModal} | ||||
|         onConfirm={confirmDelete} | ||||
|         title="Delete Category" | ||||
|         message={`Are you sure you want to delete "${deletingCategory?.name}"? This action cannot be undone.`} | ||||
|         isLoading={deleteLoading} | ||||
|       /> | ||||
|     </div> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| @ -2,7 +2,6 @@ import React, { useState, useEffect } from 'react'; | ||||
| import { Grocery } from '../types'; | ||||
| import { groceryApi } from '../services/api'; | ||||
| import AddGroceryModal from './AddGroceryModal'; | ||||
| import ConfirmDeleteModal from './ConfirmDeleteModal'; | ||||
| 
 | ||||
| const GroceryList: React.FC = () => { | ||||
|   const [groceries, setGroceries] = useState<Grocery[]>([]); | ||||
| @ -10,10 +9,6 @@ const GroceryList: React.FC = () => { | ||||
|   const [message, setMessage] = useState(''); | ||||
|   const [isModalOpen, setIsModalOpen] = useState(false); | ||||
|   const [editingGrocery, setEditingGrocery] = useState<Grocery | null>(null); | ||||
|   const [deletingGrocery, setDeletingGrocery] = useState<Grocery | null>(null); | ||||
|   const [deleteLoading, setDeleteLoading] = useState(false); | ||||
|   const [sortField, setSortField] = useState<string>('name'); | ||||
|   const [sortDirection, setSortDirection] = useState<'asc' | 'desc'>('asc'); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchGroceries(); | ||||
| @ -32,37 +27,25 @@ const GroceryList: React.FC = () => { | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const handleDelete = async (grocery: Grocery) => { | ||||
|     setDeletingGrocery(grocery); | ||||
|   }; | ||||
| 
 | ||||
|   const confirmDelete = async () => { | ||||
|     if (!deletingGrocery) return; | ||||
| 
 | ||||
|     try { | ||||
|       setDeleteLoading(true); | ||||
|       await groceryApi.delete(deletingGrocery.id); | ||||
|       setMessage('Grocery deleted successfully!'); | ||||
|       setDeletingGrocery(null); | ||||
|       fetchGroceries(); | ||||
|       setTimeout(() => setMessage(''), 1500); | ||||
|     } catch (error: any) { | ||||
|       console.error('Error deleting grocery:', error); | ||||
|       if (error.response?.status === 400) { | ||||
|         setMessage('Cannot delete grocery: products are still associated with this grocery.'); | ||||
|       } else { | ||||
|         setMessage('Error deleting grocery. Please try again.'); | ||||
|   const handleDelete = async (id: number) => { | ||||
|     if (window.confirm('Are you sure you want to delete this grocery?')) { | ||||
|       try { | ||||
|         await groceryApi.delete(id); | ||||
|         setMessage('Grocery deleted successfully!'); | ||||
|         fetchGroceries(); | ||||
|         setTimeout(() => setMessage(''), 3000); | ||||
|       } catch (error: any) { | ||||
|         console.error('Error deleting grocery:', error); | ||||
|         if (error.response?.status === 400) { | ||||
|           setMessage('Cannot delete grocery: products are still associated with this grocery.'); | ||||
|         } else { | ||||
|           setMessage('Error deleting grocery. Please try again.'); | ||||
|         } | ||||
|         setTimeout(() => setMessage(''), 5000); | ||||
|       } | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } finally { | ||||
|       setDeleteLoading(false); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const handleCloseDeleteModal = () => { | ||||
|     setDeletingGrocery(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleEdit = (grocery: Grocery) => { | ||||
|     setEditingGrocery(grocery); | ||||
|     setIsModalOpen(true); | ||||
| @ -74,76 +57,6 @@ const GroceryList: React.FC = () => { | ||||
|     fetchGroceries(); | ||||
|   }; | ||||
| 
 | ||||
|   const handleSort = (field: string) => { | ||||
|     if (field === sortField) { | ||||
|       setSortDirection(sortDirection === 'asc' ? 'desc' : 'asc'); | ||||
|     } else { | ||||
|       setSortField(field); | ||||
|       setSortDirection('asc'); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const sortedGroceries = [...groceries].sort((a, b) => { | ||||
|     let aValue: any; | ||||
|     let bValue: any; | ||||
| 
 | ||||
|     switch (sortField) { | ||||
|       case 'name': | ||||
|         aValue = a.name; | ||||
|         bValue = b.name; | ||||
|         break; | ||||
|       case 'category': | ||||
|         aValue = a.category.name; | ||||
|         bValue = b.category.name; | ||||
|         break; | ||||
|       case 'created_at': | ||||
|         aValue = a.created_at; | ||||
|         bValue = b.created_at; | ||||
|         break; | ||||
|       default: | ||||
|         aValue = ''; | ||||
|         bValue = ''; | ||||
|     } | ||||
| 
 | ||||
|     // Handle null/undefined values
 | ||||
|     if (aValue === null || aValue === undefined) aValue = ''; | ||||
|     if (bValue === null || bValue === undefined) bValue = ''; | ||||
| 
 | ||||
|     // Convert to string for comparison
 | ||||
|     const aStr = String(aValue).toLowerCase(); | ||||
|     const bStr = String(bValue).toLowerCase(); | ||||
| 
 | ||||
|     if (sortDirection === 'asc') { | ||||
|       return aStr.localeCompare(bStr); | ||||
|     } else { | ||||
|       return bStr.localeCompare(aStr); | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const getSortIcon = (field: string) => { | ||||
|     if (sortField !== field) { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 16V4m0 0L3 8m4-4l4 4m6 0v12m0 0l4-4m-4 4l-4-4" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|      | ||||
|     if (sortDirection === 'asc') { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 15l7-7 7 7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } else { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   if (loading) { | ||||
|     return ( | ||||
|       <div className="flex justify-center items-center h-64"> | ||||
| @ -153,105 +66,87 @@ const GroceryList: React.FC = () => { | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <div className="space-y-6"> | ||||
|       <div className="flex justify-between items-center"> | ||||
|         <h1 className="text-2xl font-bold text-gray-900">Groceries</h1> | ||||
|         <button | ||||
|           onClick={() => setIsModalOpen(true)} | ||||
|           className="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded" | ||||
|         > | ||||
|           Add New Grocery | ||||
|         </button> | ||||
|       </div> | ||||
| 
 | ||||
|       {message && ( | ||||
|         <div className={`px-4 py-3 rounded ${ | ||||
|           message.includes('Error') || message.includes('Cannot')  | ||||
|             ? 'bg-red-50 border border-red-200 text-red-700'  | ||||
|             : 'bg-green-50 border border-green-200 text-green-700' | ||||
|         }`}>
 | ||||
|           {message} | ||||
|         </div> | ||||
|       )} | ||||
| 
 | ||||
|       <div className="bg-white shadow rounded-lg overflow-hidden"> | ||||
|         {groceries.length === 0 ? ( | ||||
|           <div className="text-center py-12"> | ||||
|             <svg className="mx-auto h-12 w-12 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 48 48"> | ||||
|               <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M20 7l-8-4-8 4m16 0l-8 4m8-4v10l-8 4m0-10L4 7m8 4v10M4 7v10l8 4" /> | ||||
|             </svg> | ||||
|             <h3 className="mt-2 text-sm font-medium text-gray-900">No groceries</h3> | ||||
|             <p className="mt-1 text-sm text-gray-500">Get started by adding your first grocery.</p> | ||||
|     <div className="max-w-4xl mx-auto"> | ||||
|       <div className="bg-white shadow rounded-lg"> | ||||
|         <div className="px-4 py-5 sm:p-6"> | ||||
|           <div className="flex justify-between items-center mb-4"> | ||||
|             <h3 className="text-lg leading-6 font-medium text-gray-900"> | ||||
|               Groceries | ||||
|             </h3> | ||||
|             <button | ||||
|               onClick={() => setIsModalOpen(true)} | ||||
|               className="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded" | ||||
|             > | ||||
|               Add Grocery | ||||
|             </button> | ||||
|           </div> | ||||
|         ) : ( | ||||
|           <table className="min-w-full divide-y divide-gray-200"> | ||||
|             <thead className="bg-gray-50"> | ||||
|               <tr> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('name')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Name | ||||
|                     {getSortIcon('name')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('category')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Category | ||||
|                     {getSortIcon('category')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('created_at')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Created | ||||
|                     {getSortIcon('created_at')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Actions | ||||
|                 </th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody className="bg-white divide-y divide-gray-200"> | ||||
|               {sortedGroceries.map((grocery) => ( | ||||
|                 <tr key={grocery.id} className="hover:bg-gray-50"> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <div className="text-sm font-medium text-gray-900"> | ||||
|                       {grocery.name} | ||||
|                     </div> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> | ||||
|                     {grocery.category.name} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                     {new Date(grocery.created_at).toLocaleDateString()} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm font-medium"> | ||||
|                     <button | ||||
|                       onClick={() => handleEdit(grocery)} | ||||
|                       className="text-indigo-600 hover:text-indigo-900 mr-3" | ||||
|                     > | ||||
|                       Edit | ||||
|                     </button> | ||||
|                     <button | ||||
|                       onClick={() => handleDelete(grocery)} | ||||
|                       className="text-red-600 hover:text-red-900" | ||||
|                     > | ||||
|                       Delete | ||||
|                     </button> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|             </tbody> | ||||
|           </table> | ||||
|         )} | ||||
| 
 | ||||
|           {message && ( | ||||
|             <div className={`mb-4 p-4 rounded-md ${ | ||||
|               message.includes('Error') || message.includes('Cannot')  | ||||
|                 ? 'bg-red-50 text-red-700'  | ||||
|                 : 'bg-green-50 text-green-700' | ||||
|             }`}>
 | ||||
|               {message} | ||||
|             </div> | ||||
|           )} | ||||
| 
 | ||||
|           {groceries.length === 0 ? ( | ||||
|             <div className="text-center py-8"> | ||||
|               <p className="text-gray-500">No groceries found. Add your first grocery!</p> | ||||
|             </div> | ||||
|           ) : ( | ||||
|             <div className="overflow-hidden shadow ring-1 ring-black ring-opacity-5 md:rounded-lg"> | ||||
|               <table className="min-w-full divide-y divide-gray-300"> | ||||
|                 <thead className="bg-gray-50"> | ||||
|                   <tr> | ||||
|                     <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                       Name | ||||
|                     </th> | ||||
|                     <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                       Category | ||||
|                     </th> | ||||
|                     <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                       Created | ||||
|                     </th> | ||||
|                     <th className="px-6 py-3 text-right text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                       Actions | ||||
|                     </th> | ||||
|                   </tr> | ||||
|                 </thead> | ||||
|                 <tbody className="bg-white divide-y divide-gray-200"> | ||||
|                   {groceries.map((grocery) => ( | ||||
|                     <tr key={grocery.id} className="hover:bg-gray-50"> | ||||
|                       <td className="px-6 py-4 whitespace-nowrap text-sm font-medium text-gray-900"> | ||||
|                         {grocery.name} | ||||
|                       </td> | ||||
|                       <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                         {grocery.category.name} | ||||
|                       </td> | ||||
|                       <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                         {new Date(grocery.created_at).toLocaleDateString()} | ||||
|                       </td> | ||||
|                       <td className="px-6 py-4 whitespace-nowrap text-right text-sm font-medium"> | ||||
|                         <button | ||||
|                           onClick={() => handleEdit(grocery)} | ||||
|                           className="text-indigo-600 hover:text-indigo-900 mr-4" | ||||
|                         > | ||||
|                           Edit | ||||
|                         </button> | ||||
|                         <button | ||||
|                           onClick={() => handleDelete(grocery.id)} | ||||
|                           className="text-red-600 hover:text-red-900" | ||||
|                         > | ||||
|                           Delete | ||||
|                         </button> | ||||
|                       </td> | ||||
|                     </tr> | ||||
|                   ))} | ||||
|                 </tbody> | ||||
|               </table> | ||||
|             </div> | ||||
|           )} | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       {isModalOpen && ( | ||||
| @ -260,15 +155,6 @@ const GroceryList: React.FC = () => { | ||||
|           onClose={handleModalClose} | ||||
|         /> | ||||
|       )} | ||||
| 
 | ||||
|       <ConfirmDeleteModal | ||||
|         isOpen={!!deletingGrocery} | ||||
|         onClose={handleCloseDeleteModal} | ||||
|         onConfirm={confirmDelete} | ||||
|         title="Delete Grocery" | ||||
|         message={`Are you sure you want to delete "${deletingGrocery?.name}"? This action cannot be undone.`} | ||||
|         isLoading={deleteLoading} | ||||
|       /> | ||||
|     </div> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| @ -14,8 +14,6 @@ const ProductList: React.FC = () => { | ||||
|   const [editingProduct, setEditingProduct] = useState<Product | null>(null); | ||||
|   const [deletingProduct, setDeletingProduct] = useState<Product | null>(null); | ||||
|   const [deleteLoading, setDeleteLoading] = useState(false); | ||||
|   const [sortField, setSortField] = useState<string>('name'); | ||||
|   const [sortDirection, setSortDirection] = useState<'asc' | 'desc'>('asc'); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchProducts(); | ||||
| @ -79,92 +77,6 @@ const ProductList: React.FC = () => { | ||||
|     setDeletingProduct(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleSort = (field: string) => { | ||||
|     if (field === sortField) { | ||||
|       setSortDirection(sortDirection === 'asc' ? 'desc' : 'asc'); | ||||
|     } else { | ||||
|       setSortField(field); | ||||
|       setSortDirection('asc'); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const sortedProducts = [...products].sort((a, b) => { | ||||
|     let aValue: any; | ||||
|     let bValue: any; | ||||
| 
 | ||||
|     switch (sortField) { | ||||
|       case 'name': | ||||
|         aValue = a.name; | ||||
|         bValue = b.name; | ||||
|         break; | ||||
|       case 'grocery': | ||||
|         aValue = a.grocery.name; | ||||
|         bValue = b.grocery.name; | ||||
|         break; | ||||
|       case 'category': | ||||
|         aValue = a.grocery.category.name; | ||||
|         bValue = b.grocery.category.name; | ||||
|         break; | ||||
|       case 'brand': | ||||
|         aValue = a.brand?.name || ''; | ||||
|         bValue = b.brand?.name || ''; | ||||
|         break; | ||||
|       case 'weight': | ||||
|         aValue = a.weight || 0; | ||||
|         bValue = b.weight || 0; | ||||
|         break; | ||||
|       default: | ||||
|         aValue = ''; | ||||
|         bValue = ''; | ||||
|     } | ||||
| 
 | ||||
|     // Handle null/undefined values
 | ||||
|     if (aValue === null || aValue === undefined) aValue = ''; | ||||
|     if (bValue === null || bValue === undefined) bValue = ''; | ||||
| 
 | ||||
|     // Convert to string for comparison (except for numbers)
 | ||||
|     if (typeof aValue === 'number' && typeof bValue === 'number') { | ||||
|       if (sortDirection === 'asc') { | ||||
|         return aValue - bValue; | ||||
|       } else { | ||||
|         return bValue - aValue; | ||||
|       } | ||||
|     } else { | ||||
|       const aStr = String(aValue).toLowerCase(); | ||||
|       const bStr = String(bValue).toLowerCase(); | ||||
| 
 | ||||
|       if (sortDirection === 'asc') { | ||||
|         return aStr.localeCompare(bStr); | ||||
|       } else { | ||||
|         return bStr.localeCompare(aStr); | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const getSortIcon = (field: string) => { | ||||
|     if (sortField !== field) { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 16V4m0 0L3 8m4-4l4 4m6 0v12m0 0l4-4m-4 4l-4-4" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|      | ||||
|     if (sortDirection === 'asc') { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 15l7-7 7 7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } else { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   if (loading) { | ||||
|     return ( | ||||
|       <div className="flex justify-center items-center h-64"> | ||||
| @ -207,50 +119,20 @@ const ProductList: React.FC = () => { | ||||
|           <table className="min-w-full divide-y divide-gray-200"> | ||||
|             <thead className="bg-gray-50"> | ||||
|               <tr> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('name')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Name | ||||
|                     {getSortIcon('name')} | ||||
|                   </div> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Name | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('grocery')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Grocery | ||||
|                     {getSortIcon('grocery')} | ||||
|                   </div> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Grocery | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('category')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Category | ||||
|                     {getSortIcon('category')} | ||||
|                   </div> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Brand | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('brand')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Brand | ||||
|                     {getSortIcon('brand')} | ||||
|                   </div> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Weight | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('weight')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Weight | ||||
|                     {getSortIcon('weight')} | ||||
|                   </div> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Organic | ||||
|                 </th> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Actions | ||||
| @ -258,7 +140,7 @@ const ProductList: React.FC = () => { | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody className="bg-white divide-y divide-gray-200"> | ||||
|               {sortedProducts.map((product) => ( | ||||
|               {products.map((product) => ( | ||||
|                 <tr key={product.id} className="hover:bg-gray-50"> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <div className="text-sm font-medium text-gray-900"> | ||||
| @ -267,9 +149,7 @@ const ProductList: React.FC = () => { | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <div className="text-sm text-gray-900">{product.grocery.name}</div> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> | ||||
|                     {product.grocery.category.name} | ||||
|                     <div className="text-xs text-gray-500">{product.grocery.category.name}</div> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> | ||||
|                     {product.brand ? product.brand.name : '-'} | ||||
| @ -277,6 +157,15 @@ const ProductList: React.FC = () => { | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> | ||||
|                     {product.weight ? `${product.weight}${product.weight_unit}` : '-'} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <span className={`px-2 inline-flex text-xs leading-5 font-semibold rounded-full ${ | ||||
|                       product.organic  | ||||
|                         ? 'bg-green-100 text-green-800'  | ||||
|                         : 'bg-gray-100 text-gray-800' | ||||
|                     }`}>
 | ||||
|                       {product.organic ? 'Organic' : 'Conventional'} | ||||
|                     </span> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm font-medium"> | ||||
|                     <button  | ||||
|                       onClick={() => handleEdit(product)} | ||||
|  | ||||
| @ -1,7 +1,7 @@ | ||||
| import React, { useState, useEffect } from 'react'; | ||||
| import { useSearchParams } from 'react-router-dom'; | ||||
| import { Shop, BrandInShop } from '../types'; | ||||
| import { shopApi, brandInShopApi } from '../services/api'; | ||||
| import { Shop } from '../types'; | ||||
| import { shopApi } from '../services/api'; | ||||
| import AddShopModal from './AddShopModal'; | ||||
| import ConfirmDeleteModal from './ConfirmDeleteModal'; | ||||
| 
 | ||||
| @ -14,12 +14,6 @@ const ShopList: React.FC = () => { | ||||
|   const [editingShop, setEditingShop] = useState<Shop | null>(null); | ||||
|   const [deletingShop, setDeletingShop] = useState<Shop | null>(null); | ||||
|   const [deleteLoading, setDeleteLoading] = useState(false); | ||||
|   const [sortField, setSortField] = useState<keyof Shop>('name'); | ||||
|   const [sortDirection, setSortDirection] = useState<'asc' | 'desc'>('asc'); | ||||
|   const [hoveredShop, setHoveredShop] = useState<Shop | null>(null); | ||||
|   const [showBrandsPopup, setShowBrandsPopup] = useState(false); | ||||
|   const [popupPosition, setPopupPosition] = useState({ x: 0, y: 0 }); | ||||
|   const [shopBrands, setShopBrands] = useState<Record<number, BrandInShop[]>>({}); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchShops(); | ||||
| @ -32,35 +26,11 @@ const ShopList: React.FC = () => { | ||||
|     } | ||||
|   }, [searchParams, setSearchParams]); | ||||
| 
 | ||||
|   // Handle clicking outside popup to close it
 | ||||
|   useEffect(() => { | ||||
|     const handleClickOutside = (event: MouseEvent) => { | ||||
|       const target = event.target as HTMLElement; | ||||
|       if (showBrandsPopup && !target.closest('.brands-popup') && !target.closest('.brands-cell')) { | ||||
|         setShowBrandsPopup(false); | ||||
|         setHoveredShop(null); | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     if (showBrandsPopup) { | ||||
|       document.addEventListener('mousedown', handleClickOutside); | ||||
|     } | ||||
| 
 | ||||
|     return () => { | ||||
|       document.removeEventListener('mousedown', handleClickOutside); | ||||
|     }; | ||||
|   }, [showBrandsPopup]); | ||||
| 
 | ||||
|   const fetchShops = async () => { | ||||
|     try { | ||||
|       setLoading(true); | ||||
|       const response = await shopApi.getAll(); | ||||
|       setShops(response.data); | ||||
|        | ||||
|       // Load brands for all shops
 | ||||
|       for (const shop of response.data) { | ||||
|         loadShopBrands(shop.id); | ||||
|       } | ||||
|     } catch (err) { | ||||
|       setError('Failed to fetch shops'); | ||||
|       console.error('Error fetching shops:', err); | ||||
| @ -69,18 +39,6 @@ const ShopList: React.FC = () => { | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const loadShopBrands = async (shopId: number) => { | ||||
|     try { | ||||
|       const response = await brandInShopApi.getByShop(shopId); | ||||
|       setShopBrands(prev => ({ | ||||
|         ...prev, | ||||
|         [shopId]: response.data | ||||
|       })); | ||||
|     } catch (err) { | ||||
|       console.error('Error loading shop brands:', err); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const handleShopAdded = () => { | ||||
|     fetchShops(); // Refresh the shops list
 | ||||
|   }; | ||||
| @ -119,118 +77,6 @@ const ShopList: React.FC = () => { | ||||
|     setDeletingShop(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleBrandsHover = (shop: Shop, mouseEvent: React.MouseEvent) => { | ||||
|     const brands = shopBrands[shop.id] || []; | ||||
|     if (brands.length === 0) return; | ||||
|      | ||||
|     const rect = mouseEvent.currentTarget.getBoundingClientRect(); | ||||
|     const popupWidth = 300; | ||||
|     const popupHeight = 200; | ||||
|      | ||||
|     let x = mouseEvent.clientX + 10; | ||||
|     let y = mouseEvent.clientY - 10; | ||||
|      | ||||
|     // Adjust if popup would go off screen
 | ||||
|     if (x + popupWidth > window.innerWidth) { | ||||
|       x = mouseEvent.clientX - popupWidth - 10; | ||||
|     } | ||||
|     if (y + popupHeight > window.innerHeight) { | ||||
|       y = mouseEvent.clientY - popupHeight + 10; | ||||
|     } | ||||
|     if (y < 0) { | ||||
|       y = 10; | ||||
|     } | ||||
|      | ||||
|     setHoveredShop(shop); | ||||
|     setPopupPosition({ x, y }); | ||||
|     setShowBrandsPopup(true); | ||||
|   }; | ||||
| 
 | ||||
|   const handleBrandsLeave = () => { | ||||
|     setShowBrandsPopup(false); | ||||
|     setHoveredShop(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleBrandsClick = (shop: Shop, mouseEvent: React.MouseEvent) => { | ||||
|     const brands = shopBrands[shop.id] || []; | ||||
|     if (brands.length === 0) return; | ||||
|      | ||||
|     mouseEvent.stopPropagation(); | ||||
|     const rect = mouseEvent.currentTarget.getBoundingClientRect(); | ||||
|     const popupWidth = 300; | ||||
|     const popupHeight = 200; | ||||
|      | ||||
|     let x = mouseEvent.clientX + 10; | ||||
|     let y = mouseEvent.clientY - 10; | ||||
|      | ||||
|     // Adjust if popup would go off screen
 | ||||
|     if (x + popupWidth > window.innerWidth) { | ||||
|       x = mouseEvent.clientX - popupWidth - 10; | ||||
|     } | ||||
|     if (y + popupHeight > window.innerHeight) { | ||||
|       y = mouseEvent.clientY - popupHeight + 10; | ||||
|     } | ||||
|     if (y < 0) { | ||||
|       y = 10; | ||||
|     } | ||||
|      | ||||
|     setHoveredShop(shop); | ||||
|     setPopupPosition({ x, y }); | ||||
|     setShowBrandsPopup(true); | ||||
|   }; | ||||
| 
 | ||||
|   const handleSort = (field: keyof Shop) => { | ||||
|     if (field === sortField) { | ||||
|       setSortDirection(sortDirection === 'asc' ? 'desc' : 'asc'); | ||||
|     } else { | ||||
|       setSortField(field); | ||||
|       setSortDirection('asc'); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const sortedShops = [...shops].sort((a, b) => { | ||||
|     let aValue = a[sortField]; | ||||
|     let bValue = b[sortField]; | ||||
| 
 | ||||
|     // Handle null/undefined values
 | ||||
|     if (aValue === null || aValue === undefined) aValue = ''; | ||||
|     if (bValue === null || bValue === undefined) bValue = ''; | ||||
| 
 | ||||
|     // Convert to string for comparison
 | ||||
|     const aStr = String(aValue).toLowerCase(); | ||||
|     const bStr = String(bValue).toLowerCase(); | ||||
| 
 | ||||
|     if (sortDirection === 'asc') { | ||||
|       return aStr.localeCompare(bStr); | ||||
|     } else { | ||||
|       return bStr.localeCompare(aStr); | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const getSortIcon = (field: keyof Shop) => { | ||||
|     if (sortField !== field) { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 16V4m0 0L3 8m4-4l4 4m6 0v12m0 0l4-4m-4 4l-4-4" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|      | ||||
|     if (sortDirection === 'asc') { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 15l7-7 7 7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } else { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   if (loading) { | ||||
|     return ( | ||||
|       <div className="flex justify-center items-center h-64"> | ||||
| @ -267,110 +113,64 @@ const ShopList: React.FC = () => { | ||||
|             <p className="mt-1 text-sm text-gray-500">Get started by adding your first shop.</p> | ||||
|           </div> | ||||
|         ) : ( | ||||
|           <table className="min-w-full divide-y divide-gray-200"> | ||||
|             <thead className="bg-gray-50"> | ||||
|               <tr> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('name')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Name | ||||
|                     {getSortIcon('name')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('city')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     City | ||||
|                     {getSortIcon('city')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('address')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Address | ||||
|                     {getSortIcon('address')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Brands | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('created_at')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Created | ||||
|                     {getSortIcon('created_at')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Actions | ||||
|                 </th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody className="bg-white divide-y divide-gray-200"> | ||||
|               {sortedShops.map((shop) => ( | ||||
|                 <tr key={shop.id} className="hover:bg-gray-50"> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <div className="text-sm font-medium text-gray-900"> | ||||
|                       {shop.name} | ||||
|                     </div> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> | ||||
|                     {shop.city} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> | ||||
|                     {shop.address || '-'} | ||||
|                   </td> | ||||
|                   <td  | ||||
|                     className={`brands-cell px-6 py-4 whitespace-nowrap text-sm ${ | ||||
|                       (shopBrands[shop.id]?.length || 0) > 0  | ||||
|                         ? 'text-blue-600 hover:text-blue-800 cursor-pointer hover:bg-blue-50'  | ||||
|                         : 'text-gray-900' | ||||
|                     }`}
 | ||||
|                     onMouseEnter={(e) => handleBrandsHover(shop, e)} | ||||
|                     onMouseLeave={handleBrandsLeave} | ||||
|                     onClick={(e) => handleBrandsClick(shop, e)} | ||||
|                     title={(shopBrands[shop.id]?.length || 0) > 0 ? 'Click to view brands' : ''} | ||||
|                   > | ||||
|                     {(shopBrands[shop.id]?.length || 0) > 0 ? ( | ||||
|                       <> | ||||
|                         {(shopBrands[shop.id]?.length || 0)} brand{(shopBrands[shop.id]?.length || 0) !== 1 ? 's' : ''} | ||||
|                         <svg className="inline-block w-4 h-4 ml-1" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z" /> | ||||
|                         </svg> | ||||
|                       </> | ||||
|                     ) : ( | ||||
|                       '-' | ||||
|                     )} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                     {new Date(shop.created_at).toLocaleDateString()} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm font-medium"> | ||||
|           <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6 p-6"> | ||||
|             {shops.map((shop) => ( | ||||
|               <div key={shop.id} className="bg-white border border-gray-200 rounded-lg p-6 hover:shadow-md transition-shadow"> | ||||
|                 <div className="flex items-center justify-between mb-4"> | ||||
|                   <h3 className="text-lg font-medium text-gray-900">{shop.name}</h3> | ||||
|                   <div className="flex space-x-2"> | ||||
|                     <button  | ||||
|                       onClick={() => handleEditShop(shop)} | ||||
|                       className="text-indigo-600 hover:text-indigo-900 mr-3" | ||||
|                       className="text-indigo-600 hover:text-indigo-900 text-sm" | ||||
|                     > | ||||
|                       Edit | ||||
|                     </button> | ||||
|                     <button  | ||||
|                       onClick={() => handleDeleteShop(shop)} | ||||
|                       className="text-red-600 hover:text-red-900" | ||||
|                       className="text-red-600 hover:text-red-900 text-sm" | ||||
|                     > | ||||
|                       Delete | ||||
|                     </button> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|             </tbody> | ||||
|           </table> | ||||
|                   </div> | ||||
|                 </div> | ||||
|                  | ||||
|                 <div className="space-y-2"> | ||||
|                   <div className="flex items-center text-sm text-gray-600"> | ||||
|                     <svg className="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                       <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z" /> | ||||
|                       <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M15 11a3 3 0 11-6 0 3 3 0 016 0z" /> | ||||
|                     </svg> | ||||
|                     {shop.city} | ||||
|                   </div> | ||||
|                    | ||||
|                   {shop.address && ( | ||||
|                     <div className="flex items-start text-sm text-gray-600"> | ||||
|                       <svg className="w-4 h-4 mr-2 mt-0.5" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                         <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M3 8l7.89 7.89a2 2 0 002.83 0L21 8M5 19h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z" /> | ||||
|                       </svg> | ||||
|                       {shop.address} | ||||
|                     </div> | ||||
|                   )} | ||||
|                    | ||||
|                   <div className="flex items-center text-sm text-gray-600"> | ||||
|                     <svg className="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                       <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z" /> | ||||
|                     </svg> | ||||
|                     Added {new Date(shop.created_at).toLocaleDateString()} | ||||
|                   </div> | ||||
|                    | ||||
|                   {shop.updated_at && ( | ||||
|                     <div className="flex items-center text-sm text-gray-600"> | ||||
|                       <svg className="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                         <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15" /> | ||||
|                       </svg> | ||||
|                       Updated {new Date(shop.updated_at).toLocaleDateString()} | ||||
|                     </div> | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|             ))} | ||||
|           </div> | ||||
|         )} | ||||
|       </div> | ||||
| 
 | ||||
| @ -389,32 +189,6 @@ const ShopList: React.FC = () => { | ||||
|         message={`Are you sure you want to delete "${deletingShop?.name}"? This action cannot be undone.`} | ||||
|         isLoading={deleteLoading} | ||||
|       /> | ||||
| 
 | ||||
|       {/* Brands Popup */} | ||||
|       {showBrandsPopup && hoveredShop && (shopBrands[hoveredShop.id]?.length || 0) > 0 && ( | ||||
|         <div  | ||||
|           className="brands-popup fixed z-50 bg-white border border-gray-200 rounded-lg shadow-lg p-4 max-w-sm" | ||||
|           style={{ | ||||
|             left: `${popupPosition.x}px`, | ||||
|             top: `${popupPosition.y}px`, | ||||
|             maxHeight: '200px', | ||||
|             overflowY: 'auto' | ||||
|           }} | ||||
|           onMouseEnter={() => setShowBrandsPopup(true)} | ||||
|           onMouseLeave={handleBrandsLeave} | ||||
|         > | ||||
|           <div className="space-y-2"> | ||||
|             <h4 className="font-medium text-gray-900 mb-2">Available Brands:</h4> | ||||
|             {shopBrands[hoveredShop.id]?.map((brandInShop, index) => ( | ||||
|               <div key={index} className="border-b border-gray-100 pb-1 last:border-b-0"> | ||||
|                 <div className="text-sm font-medium text-gray-900"> | ||||
|                   {brandInShop.brand.name} | ||||
|                 </div> | ||||
|               </div> | ||||
|             ))} | ||||
|           </div> | ||||
|         </div> | ||||
|       )} | ||||
|     </div> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| @ -1,27 +1,18 @@ | ||||
| import React, { useState, useEffect, useCallback } from 'react'; | ||||
| import { Shop, Product, ShoppingEventCreate, ProductInEvent, ShoppingEvent, BrandInShop } from '../types'; | ||||
| import { shopApi, productApi, shoppingEventApi, brandInShopApi } from '../services/api'; | ||||
| import { useParams, useNavigate } from 'react-router-dom'; | ||||
| import { Shop, Product, ShoppingEventCreate, ProductInEvent } from '../types'; | ||||
| import { shopApi, productApi, shoppingEventApi } from '../services/api'; | ||||
| 
 | ||||
| interface AddShoppingEventModalProps { | ||||
|   isOpen: boolean; | ||||
|   onClose: () => void; | ||||
|   onEventAdded: () => void; | ||||
|   editEvent?: ShoppingEvent | null; | ||||
| } | ||||
| 
 | ||||
| const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({  | ||||
|   isOpen,  | ||||
|   onClose,  | ||||
|   onEventAdded,  | ||||
|   editEvent  | ||||
| }) => { | ||||
| const ShoppingEventForm: React.FC = () => { | ||||
|   const { id } = useParams<{ id: string }>(); | ||||
|   const navigate = useNavigate(); | ||||
|   const [shops, setShops] = useState<Shop[]>([]); | ||||
|   const [products, setProducts] = useState<Product[]>([]); | ||||
|   const [shopBrands, setShopBrands] = useState<BrandInShop[]>([]); | ||||
|   const [loading, setLoading] = useState(false); | ||||
|   const [loadingEvent, setLoadingEvent] = useState(false); | ||||
|   const [message, setMessage] = useState(''); | ||||
|    | ||||
|   const isEditMode = Boolean(editEvent); | ||||
|   const isEditMode = Boolean(id); | ||||
|    | ||||
|   const [formData, setFormData] = useState<ShoppingEventCreate>({ | ||||
|     shop_id: 0, | ||||
| @ -45,17 +36,21 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|     return Math.round(total * 100) / 100; // Round to 2 decimal places to avoid floating-point errors
 | ||||
|   }; | ||||
| 
 | ||||
|   const loadEventData = useCallback(() => { | ||||
|     if (editEvent) { | ||||
|   const fetchShoppingEvent = useCallback(async (eventId: number) => { | ||||
|     try { | ||||
|       setLoadingEvent(true); | ||||
|       const response = await shoppingEventApi.getById(eventId); | ||||
|       const event = response.data; | ||||
|        | ||||
|       // Use the date directly if it's already in YYYY-MM-DD format, otherwise format it
 | ||||
|       let formattedDate = editEvent.date; | ||||
|       if (editEvent.date.includes('T') || editEvent.date.length > 10) { | ||||
|       let formattedDate = event.date; | ||||
|       if (event.date.includes('T') || event.date.length > 10) { | ||||
|         // If the date includes time or is longer than YYYY-MM-DD, extract just the date part
 | ||||
|         formattedDate = editEvent.date.split('T')[0]; | ||||
|         formattedDate = event.date.split('T')[0]; | ||||
|       } | ||||
|        | ||||
|       // Map products to the format we need
 | ||||
|       const mappedProducts = editEvent.products.map(p => ({ | ||||
|       const mappedProducts = event.products.map(p => ({ | ||||
|         product_id: p.id, | ||||
|         amount: p.amount, | ||||
|         price: p.price | ||||
| @ -65,68 +60,34 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|       const calculatedTotal = calculateTotal(mappedProducts); | ||||
|        | ||||
|       // Check if existing total matches calculated total (with small tolerance for floating point)
 | ||||
|       const existingTotal = editEvent.total_amount || 0; | ||||
|       const existingTotal = event.total_amount || 0; | ||||
|       const totalMatches = Math.abs(existingTotal - calculatedTotal) < 0.01; | ||||
|        | ||||
|       setFormData({ | ||||
|         shop_id: editEvent.shop.id, | ||||
|         shop_id: event.shop.id, | ||||
|         date: formattedDate, | ||||
|         total_amount: editEvent.total_amount, | ||||
|         notes: editEvent.notes || '', | ||||
|         total_amount: event.total_amount, | ||||
|         notes: event.notes || '', | ||||
|         products: [] | ||||
|       }); | ||||
|        | ||||
|       setSelectedProducts(mappedProducts); | ||||
|       setAutoCalculate(totalMatches); // Enable auto-calc if totals match, disable if they don't
 | ||||
|     } else { | ||||
|       // Reset form for adding new event
 | ||||
|       setFormData({ | ||||
|         shop_id: 0, | ||||
|         date: new Date().toISOString().split('T')[0], | ||||
|         total_amount: undefined, | ||||
|         notes: '', | ||||
|         products: [] | ||||
|       }); | ||||
|       setSelectedProducts([]); | ||||
|       setAutoCalculate(true); | ||||
|     } catch (error) { | ||||
|       console.error('Error fetching shopping event:', error); | ||||
|       setMessage('Error loading shopping event. Please try again.'); | ||||
|     } finally { | ||||
|       setLoadingEvent(false); | ||||
|     } | ||||
|     setMessage(''); | ||||
|   }, [editEvent]); | ||||
|   }, []); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (isOpen) { | ||||
|       fetchShops(); | ||||
|       fetchProducts(); | ||||
|       loadEventData(); | ||||
|     fetchShops(); | ||||
|     fetchProducts(); | ||||
|     if (isEditMode && id) { | ||||
|       fetchShoppingEvent(parseInt(id)); | ||||
|     } | ||||
|   }, [isOpen, loadEventData]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleKeyDown = (event: KeyboardEvent) => { | ||||
|       if (!isOpen) return; | ||||
|        | ||||
|       if (event.key === 'Escape') { | ||||
|         onClose(); | ||||
|       } else if (event.key === 'Enter' && !event.shiftKey && !loading) { | ||||
|         // Only trigger submit if not in a textarea and form is valid
 | ||||
|         const target = event.target as HTMLElement; | ||||
|         if (target.tagName !== 'TEXTAREA') { | ||||
|           event.preventDefault(); | ||||
|           if (formData.shop_id > 0 && selectedProducts.length > 0) { | ||||
|             handleSubmit(event as any); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     if (isOpen) { | ||||
|       document.addEventListener('keydown', handleKeyDown); | ||||
|     } | ||||
|      | ||||
|     return () => { | ||||
|       document.removeEventListener('keydown', handleKeyDown); | ||||
|     }; | ||||
|   }, [isOpen, formData, selectedProducts, loading, onClose]); | ||||
|   }, [id, isEditMode, fetchShoppingEvent]); | ||||
| 
 | ||||
|   // Update total amount whenever selectedProducts changes
 | ||||
|   useEffect(() => { | ||||
| @ -145,8 +106,6 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|       setShops(response.data); | ||||
|     } catch (error) { | ||||
|       console.error('Error fetching shops:', error); | ||||
|       setMessage('Error loading shops. Please try again.'); | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
| @ -156,35 +115,9 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|       setProducts(response.data); | ||||
|     } catch (error) { | ||||
|       console.error('Error fetching products:', error); | ||||
|       setMessage('Error loading products. Please try again.'); | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const fetchShopBrands = async (shopId: number) => { | ||||
|     if (shopId === 0) { | ||||
|       setShopBrands([]); | ||||
|       return; | ||||
|     } | ||||
|      | ||||
|     try { | ||||
|       const response = await brandInShopApi.getByShop(shopId); | ||||
|       setShopBrands(response.data); | ||||
|     } catch (error) { | ||||
|       console.error('Error fetching shop brands:', error); | ||||
|       setShopBrands([]); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   // Effect to load shop brands when shop selection changes
 | ||||
|   useEffect(() => { | ||||
|     if (formData.shop_id > 0) { | ||||
|       fetchShopBrands(formData.shop_id); | ||||
|     } else { | ||||
|       setShopBrands([]); | ||||
|     } | ||||
|   }, [formData.shop_id]); | ||||
| 
 | ||||
|   const addProductToEvent = () => { | ||||
|     if (newProductItem.product_id > 0 && newProductItem.amount > 0 && newProductItem.price >= 0) { | ||||
|       setSelectedProducts([...selectedProducts, { ...newProductItem }]); | ||||
| @ -219,22 +152,34 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|         products: selectedProducts | ||||
|       }; | ||||
| 
 | ||||
|       if (isEditMode && editEvent) { | ||||
|         await shoppingEventApi.update(editEvent.id, eventData); | ||||
|       if (isEditMode) { | ||||
|         // Update existing event
 | ||||
|         console.log('Updating event data:', eventData); | ||||
|         await shoppingEventApi.update(parseInt(id!), eventData); | ||||
|         setMessage('Shopping event updated successfully!'); | ||||
|          | ||||
|         // Navigate back to shopping events list after a short delay
 | ||||
|         setTimeout(() => { | ||||
|           navigate('/shopping-events'); | ||||
|         }, 1500); | ||||
|       } else { | ||||
|         // Create new event
 | ||||
|         await shoppingEventApi.create(eventData); | ||||
|         setMessage('Shopping event created successfully!'); | ||||
|          | ||||
|         // Reset form for add mode
 | ||||
|         setFormData({ | ||||
|           shop_id: 0, | ||||
|           date: new Date().toISOString().split('T')[0], | ||||
|           total_amount: undefined, | ||||
|           notes: '', | ||||
|           products: [] | ||||
|         }); | ||||
|         setSelectedProducts([]); | ||||
|       } | ||||
|        | ||||
|       setTimeout(() => { | ||||
|         onEventAdded(); | ||||
|         onClose(); | ||||
|       }, 1500); | ||||
|     } catch (error) { | ||||
|       console.error('Error saving shopping event:', error); | ||||
|       console.error('Full error object:', error); | ||||
|       setMessage(`Error ${isEditMode ? 'updating' : 'creating'} shopping event. Please try again.`); | ||||
|       setTimeout(() => setMessage(''), 3000); | ||||
|     } finally { | ||||
|       setLoading(false); | ||||
|     } | ||||
| @ -249,48 +194,37 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|     return `${product.name}${organicEmoji} ${weightInfo}`; | ||||
|   }; | ||||
| 
 | ||||
|   // Filter products based on selected shop's brands
 | ||||
|   const getFilteredProducts = () => { | ||||
|     // If no shop is selected or shop has no brands, show all products
 | ||||
|     if (formData.shop_id === 0 || shopBrands.length === 0) { | ||||
|       return products; | ||||
|     } | ||||
|      | ||||
|     // Get brand IDs available in the selected shop
 | ||||
|     const availableBrandIds = shopBrands.map(sb => sb.brand_id); | ||||
|      | ||||
|     // Filter products to only show those with brands available in the shop
 | ||||
|     // Also include products without brands (brand_id is null/undefined)
 | ||||
|     return products.filter(product =>  | ||||
|       !product.brand_id || availableBrandIds.includes(product.brand_id) | ||||
|   if (loadingEvent) { | ||||
|     return ( | ||||
|       <div className="flex justify-center items-center h-64"> | ||||
|         <div className="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-500"></div> | ||||
|       </div> | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   if (!isOpen) return null; | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <div className="fixed inset-0 bg-gray-600 bg-opacity-50 overflow-y-auto h-full w-full z-50"> | ||||
|       <div className="relative top-10 mx-auto p-5 border w-full max-w-4xl shadow-lg rounded-md bg-white"> | ||||
|         <div className="mt-3"> | ||||
|     <div className="max-w-4xl mx-auto"> | ||||
|       <div className="bg-white shadow rounded-lg"> | ||||
|         <div className="px-4 py-5 sm:p-6"> | ||||
|           <div className="flex justify-between items-center mb-4"> | ||||
|             <h3 className="text-lg font-medium text-gray-900"> | ||||
|               {isEditMode ? 'Edit Shopping Event' : 'Add New Shopping Event'} | ||||
|             <h3 className="text-lg leading-6 font-medium text-gray-900"> | ||||
|               {isEditMode ? 'Edit Shopping Event' : 'Add New Event'} | ||||
|             </h3> | ||||
|             <button | ||||
|               onClick={onClose} | ||||
|               className="text-gray-400 hover:text-gray-600" | ||||
|             > | ||||
|               <svg className="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                 <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" /> | ||||
|               </svg> | ||||
|             </button> | ||||
|             {isEditMode && ( | ||||
|               <button | ||||
|                 onClick={() => navigate('/shopping-events')} | ||||
|                 className="text-gray-500 hover:text-gray-700" | ||||
|               > | ||||
|                 ← Back to Shopping Events | ||||
|               </button> | ||||
|             )} | ||||
|           </div> | ||||
| 
 | ||||
|           {message && ( | ||||
|             <div className={`mb-4 px-4 py-3 rounded ${ | ||||
|             <div className={`mb-4 p-4 rounded-md ${ | ||||
|               message.includes('Error')  | ||||
|                 ? 'bg-red-50 border border-red-200 text-red-700'  | ||||
|                 : 'bg-green-50 border border-green-200 text-green-700' | ||||
|                 ? 'bg-red-50 text-red-700'  | ||||
|                 : 'bg-green-50 text-green-700' | ||||
|             }`}>
 | ||||
|               {message} | ||||
|             </div> | ||||
| @ -347,38 +281,12 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|                     className="w-full border border-gray-300 rounded-md px-3 py-2 focus:outline-none focus:ring-2 focus:ring-blue-500" | ||||
|                   > | ||||
|                     <option value={0}>Select a product</option> | ||||
|                     {Object.entries( | ||||
|                       getFilteredProducts().reduce((groups, product) => { | ||||
|                         const category = product.grocery.category.name; | ||||
|                         if (!groups[category]) { | ||||
|                           groups[category] = []; | ||||
|                         } | ||||
|                         groups[category].push(product); | ||||
|                         return groups; | ||||
|                       }, {} as Record<string, typeof products>) | ||||
|                     ) | ||||
|                     .sort(([a], [b]) => a.localeCompare(b)) | ||||
|                     .map(([category, categoryProducts]) => ( | ||||
|                       <optgroup key={category} label={category}> | ||||
|                         {categoryProducts | ||||
|                           .sort((a, b) => a.name.localeCompare(b.name)) | ||||
|                           .map(product => ( | ||||
|                             <option key={product.id} value={product.id}> | ||||
|                               {product.name}{product.organic ? '🌱' : ''}{product.brand ? ` (${product.brand.name})` : ''} {product.weight ? `${product.weight}${product.weight_unit}` : product.weight_unit} | ||||
|                             </option> | ||||
|                           )) | ||||
|                         } | ||||
|                       </optgroup> | ||||
|                     {products.map(product => ( | ||||
|                       <option key={product.id} value={product.id}> | ||||
|                         {product.name}{product.organic ? '🌱' : ''} ({product.grocery.category.name}) {product.weight ? `${product.weight}${product.weight_unit}` : product.weight_unit} | ||||
|                       </option> | ||||
|                     ))} | ||||
|                   </select> | ||||
|                   {formData.shop_id > 0 && ( | ||||
|                     <p className="text-xs text-gray-500 mt-1"> | ||||
|                       {shopBrands.length === 0  | ||||
|                         ? `Showing all ${products.length} products (no brand restrictions for this shop)` | ||||
|                         : `Showing ${getFilteredProducts().length} of ${products.length} products (filtered by shop's available brands)` | ||||
|                       } | ||||
|                     </p> | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div className="w-24"> | ||||
|                   <label className="block text-xs font-medium text-gray-700 mb-1"> | ||||
| @ -421,7 +329,7 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
| 
 | ||||
|               {/* Selected Products List */} | ||||
|               {selectedProducts.length > 0 && ( | ||||
|                 <div className="bg-gray-50 rounded-md p-4 max-h-40 overflow-y-auto"> | ||||
|                 <div className="bg-gray-50 rounded-md p-4"> | ||||
|                   <h4 className="font-medium text-gray-700 mb-2">Selected Items:</h4> | ||||
|                   {selectedProducts.map((item, index) => ( | ||||
|                     <div key={index} className="flex justify-between items-center py-2 border-b last:border-b-0"> | ||||
| @ -511,18 +419,24 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|             </div> | ||||
| 
 | ||||
|             {/* Submit Button */} | ||||
|             <div className="flex justify-end space-x-3 pt-4"> | ||||
|               <button | ||||
|                 type="button" | ||||
|                 onClick={onClose} | ||||
|                 className="px-4 py-2 text-sm font-medium text-gray-700 bg-gray-100 hover:bg-gray-200 rounded-md" | ||||
|               > | ||||
|                 Cancel | ||||
|               </button> | ||||
|             <div className="flex justify-end space-x-3"> | ||||
|               {isEditMode && ( | ||||
|                 <button | ||||
|                   type="button" | ||||
|                   onClick={() => navigate('/shopping-events')} | ||||
|                   className="px-4 py-2 text-sm font-medium text-gray-700 bg-gray-100 hover:bg-gray-200 rounded-md" | ||||
|                 > | ||||
|                   Cancel | ||||
|                 </button> | ||||
|               )} | ||||
|               <button | ||||
|                 type="submit" | ||||
|                 disabled={loading || formData.shop_id === 0 || selectedProducts.length === 0} | ||||
|                 className="px-4 py-2 text-sm font-medium text-white bg-blue-600 hover:bg-blue-700 rounded-md disabled:opacity-50 disabled:cursor-not-allowed" | ||||
|                 className={`px-4 py-2 text-sm font-medium text-white rounded-md disabled:opacity-50 disabled:cursor-not-allowed ${ | ||||
|                   isEditMode  | ||||
|                     ? 'bg-blue-600 hover:bg-blue-700'  | ||||
|                     : 'w-full bg-blue-500 hover:bg-blue-700 font-bold py-2 px-4 focus:outline-none focus:shadow-outline' | ||||
|                 }`}
 | ||||
|               > | ||||
|                 {loading  | ||||
|                   ? (isEditMode ? 'Updating...' : 'Creating...')  | ||||
| @ -537,4 +451,4 @@ const AddShoppingEventModal: React.FC<AddShoppingEventModalProps> = ({ | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export default AddShoppingEventModal;  | ||||
| export default ShoppingEventForm;  | ||||
| @ -1,54 +1,20 @@ | ||||
| import React, { useState, useEffect } from 'react'; | ||||
| import { useSearchParams } from 'react-router-dom'; | ||||
| import { useNavigate } from 'react-router-dom'; | ||||
| import { ShoppingEvent } from '../types'; | ||||
| import { shoppingEventApi } from '../services/api'; | ||||
| import ConfirmDeleteModal from './ConfirmDeleteModal'; | ||||
| import AddShoppingEventModal from './AddShoppingEventModal'; | ||||
| 
 | ||||
| const ShoppingEventList: React.FC = () => { | ||||
|   const [searchParams, setSearchParams] = useSearchParams(); | ||||
|   const navigate = useNavigate(); | ||||
|   const [events, setEvents] = useState<ShoppingEvent[]>([]); | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [error, setError] = useState(''); | ||||
|   const [deletingEvent, setDeletingEvent] = useState<ShoppingEvent | null>(null); | ||||
|   const [deleteLoading, setDeleteLoading] = useState(false); | ||||
|   const [isModalOpen, setIsModalOpen] = useState(false); | ||||
|   const [editingEvent, setEditingEvent] = useState<ShoppingEvent | null>(null); | ||||
|   const [hoveredEvent, setHoveredEvent] = useState<ShoppingEvent | null>(null); | ||||
|   const [showItemsPopup, setShowItemsPopup] = useState(false); | ||||
|   const [popupPosition, setPopupPosition] = useState({ x: 0, y: 0 }); | ||||
|   const [sortField, setSortField] = useState<string>('date'); | ||||
|   const [sortDirection, setSortDirection] = useState<'asc' | 'desc'>('desc'); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchEvents(); | ||||
|      | ||||
|     // Check if we should auto-open the modal
 | ||||
|     if (searchParams.get('add') === 'true') { | ||||
|       setIsModalOpen(true); | ||||
|       // Remove the parameter from URL
 | ||||
|       setSearchParams({}); | ||||
|     } | ||||
|   }, [searchParams, setSearchParams]); | ||||
| 
 | ||||
|   // Handle clicking outside popup to close it
 | ||||
|   useEffect(() => { | ||||
|     const handleClickOutside = (event: MouseEvent) => { | ||||
|       const target = event.target as HTMLElement; | ||||
|       if (showItemsPopup && !target.closest('.items-popup') && !target.closest('.items-cell')) { | ||||
|         setShowItemsPopup(false); | ||||
|         setHoveredEvent(null); | ||||
|       } | ||||
|     }; | ||||
| 
 | ||||
|     if (showItemsPopup) { | ||||
|       document.addEventListener('mousedown', handleClickOutside); | ||||
|     } | ||||
| 
 | ||||
|     return () => { | ||||
|       document.removeEventListener('mousedown', handleClickOutside); | ||||
|     }; | ||||
|   }, [showItemsPopup]); | ||||
|   }, []); | ||||
| 
 | ||||
|   const fetchEvents = async () => { | ||||
|     try { | ||||
| @ -87,167 +53,6 @@ const ShoppingEventList: React.FC = () => { | ||||
|     setDeletingEvent(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleEdit = (event: ShoppingEvent) => { | ||||
|     setEditingEvent(event); | ||||
|     setIsModalOpen(true); | ||||
|   }; | ||||
| 
 | ||||
|   const handleEventAdded = () => { | ||||
|     fetchEvents(); // Refresh the events list
 | ||||
|   }; | ||||
| 
 | ||||
|   const handleCloseModal = () => { | ||||
|     setIsModalOpen(false); | ||||
|     setEditingEvent(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleItemsHover = (event: ShoppingEvent, mouseEvent: React.MouseEvent) => { | ||||
|     if (event.products.length === 0) return; | ||||
|      | ||||
|     const rect = mouseEvent.currentTarget.getBoundingClientRect(); | ||||
|     const popupWidth = 384; // max-w-md is approximately 384px
 | ||||
|     const popupHeight = 300; // max height we set
 | ||||
|      | ||||
|     let x = mouseEvent.clientX + 10; | ||||
|     let y = mouseEvent.clientY - 10; | ||||
|      | ||||
|     // Adjust if popup would go off screen
 | ||||
|     if (x + popupWidth > window.innerWidth) { | ||||
|       x = mouseEvent.clientX - popupWidth - 10; | ||||
|     } | ||||
|     if (y + popupHeight > window.innerHeight) { | ||||
|       y = mouseEvent.clientY - popupHeight + 10; | ||||
|     } | ||||
|     if (y < 0) { | ||||
|       y = 10; | ||||
|     } | ||||
|      | ||||
|     setHoveredEvent(event); | ||||
|     setPopupPosition({ x, y }); | ||||
|     setShowItemsPopup(true); | ||||
|   }; | ||||
| 
 | ||||
|   const handleItemsLeave = () => { | ||||
|     setShowItemsPopup(false); | ||||
|     setHoveredEvent(null); | ||||
|   }; | ||||
| 
 | ||||
|   const handleItemsClick = (event: ShoppingEvent, mouseEvent: React.MouseEvent) => { | ||||
|     if (event.products.length === 0) return; | ||||
|      | ||||
|     mouseEvent.stopPropagation(); | ||||
|     const rect = mouseEvent.currentTarget.getBoundingClientRect(); | ||||
|     const popupWidth = 384; // max-w-md is approximately 384px
 | ||||
|     const popupHeight = 300; // max height we set
 | ||||
|      | ||||
|     let x = mouseEvent.clientX + 10; | ||||
|     let y = mouseEvent.clientY - 10; | ||||
|      | ||||
|     // Adjust if popup would go off screen
 | ||||
|     if (x + popupWidth > window.innerWidth) { | ||||
|       x = mouseEvent.clientX - popupWidth - 10; | ||||
|     } | ||||
|     if (y + popupHeight > window.innerHeight) { | ||||
|       y = mouseEvent.clientY - popupHeight + 10; | ||||
|     } | ||||
|     if (y < 0) { | ||||
|       y = 10; | ||||
|     } | ||||
|      | ||||
|     setHoveredEvent(event); | ||||
|     setPopupPosition({ x, y }); | ||||
|     setShowItemsPopup(true); | ||||
|   }; | ||||
| 
 | ||||
|   const handleSort = (field: string) => { | ||||
|     if (field === sortField) { | ||||
|       setSortDirection(sortDirection === 'asc' ? 'desc' : 'asc'); | ||||
|     } else { | ||||
|       setSortField(field); | ||||
|       setSortDirection('asc'); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const sortedEvents = [...events].sort((a, b) => { | ||||
|     let aValue: any; | ||||
|     let bValue: any; | ||||
| 
 | ||||
|     switch (sortField) { | ||||
|       case 'shop': | ||||
|         aValue = a.shop.name; | ||||
|         bValue = b.shop.name; | ||||
|         break; | ||||
|       case 'date': | ||||
|         aValue = new Date(a.date); | ||||
|         bValue = new Date(b.date); | ||||
|         break; | ||||
|       case 'items': | ||||
|         aValue = a.products.length; | ||||
|         bValue = b.products.length; | ||||
|         break; | ||||
|       case 'total': | ||||
|         aValue = a.total_amount || 0; | ||||
|         bValue = b.total_amount || 0; | ||||
|         break; | ||||
|       case 'notes': | ||||
|         aValue = a.notes || ''; | ||||
|         bValue = b.notes || ''; | ||||
|         break; | ||||
|       default: | ||||
|         aValue = ''; | ||||
|         bValue = ''; | ||||
|     } | ||||
| 
 | ||||
|     // Handle different data types
 | ||||
|     if (aValue instanceof Date && bValue instanceof Date) { | ||||
|       if (sortDirection === 'asc') { | ||||
|         return aValue.getTime() - bValue.getTime(); | ||||
|       } else { | ||||
|         return bValue.getTime() - aValue.getTime(); | ||||
|       } | ||||
|     } else if (typeof aValue === 'number' && typeof bValue === 'number') { | ||||
|       if (sortDirection === 'asc') { | ||||
|         return aValue - bValue; | ||||
|       } else { | ||||
|         return bValue - aValue; | ||||
|       } | ||||
|     } else { | ||||
|       // String comparison
 | ||||
|       const aStr = String(aValue).toLowerCase(); | ||||
|       const bStr = String(bValue).toLowerCase(); | ||||
| 
 | ||||
|       if (sortDirection === 'asc') { | ||||
|         return aStr.localeCompare(bStr); | ||||
|       } else { | ||||
|         return bStr.localeCompare(aStr); | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const getSortIcon = (field: string) => { | ||||
|     if (sortField !== field) { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M7 16V4m0 0L3 8m4-4l4 4m6 0v12m0 0l4-4m-4 4l-4-4" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|      | ||||
|     if (sortDirection === 'asc') { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 15l7-7 7 7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } else { | ||||
|       return ( | ||||
|         <svg className="w-4 h-4 ml-1 text-gray-600" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" /> | ||||
|         </svg> | ||||
|       ); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   if (loading) { | ||||
|     return ( | ||||
|       <div className="flex justify-center items-center h-64"> | ||||
| @ -261,7 +66,7 @@ const ShoppingEventList: React.FC = () => { | ||||
|       <div className="flex justify-between items-center"> | ||||
|         <h1 className="text-2xl font-bold text-gray-900">Shopping Events</h1> | ||||
|         <button  | ||||
|           onClick={() => setIsModalOpen(true)} | ||||
|           onClick={() => navigate('/shopping-events/new')} | ||||
|           className="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded" | ||||
|         > | ||||
|           Add New Event | ||||
| @ -284,109 +89,67 @@ const ShoppingEventList: React.FC = () => { | ||||
|             <p className="mt-1 text-sm text-gray-500">Get started by recording your first purchase.</p> | ||||
|           </div> | ||||
|         ) : ( | ||||
|           <table className="min-w-full divide-y divide-gray-200"> | ||||
|             <thead className="bg-gray-50"> | ||||
|               <tr> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('shop')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Shop | ||||
|                     {getSortIcon('shop')} | ||||
|           <div className="space-y-4 p-6"> | ||||
|             {events.map((event) => ( | ||||
|               <div key={event.id} className="border border-gray-200 rounded-lg p-6 hover:shadow-md transition-shadow"> | ||||
|                 <div className="flex justify-between items-start mb-4"> | ||||
|                   <div> | ||||
|                     <h3 className="text-lg font-medium text-gray-900">{event.shop.name}</h3> | ||||
|                     <p className="text-sm text-gray-600">{event.shop.city}</p> | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('date')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Date | ||||
|                     {getSortIcon('date')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('items')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Items | ||||
|                     {getSortIcon('items')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('total')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Total | ||||
|                     {getSortIcon('total')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th  | ||||
|                   className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider cursor-pointer hover:bg-gray-100 select-none" | ||||
|                   onClick={() => handleSort('notes')} | ||||
|                 > | ||||
|                   <div className="flex items-center"> | ||||
|                     Notes | ||||
|                     {getSortIcon('notes')} | ||||
|                   </div> | ||||
|                 </th> | ||||
|                 <th className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"> | ||||
|                   Actions | ||||
|                 </th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody className="bg-white divide-y divide-gray-200"> | ||||
|               {sortedEvents.map((event) => ( | ||||
|                 <tr key={event.id} className="hover:bg-gray-50"> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     <div className="text-sm font-medium text-gray-900">{event.shop.name}</div> | ||||
|                     <div className="text-xs text-gray-500">{event.shop.city}</div> | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-900"> | ||||
|                     {new Date(event.date).toLocaleDateString()} | ||||
|                   </td> | ||||
|                   <td  | ||||
|                     className={`items-cell px-6 py-4 whitespace-nowrap text-sm ${ | ||||
|                       event.products.length > 0  | ||||
|                         ? 'text-blue-600 hover:text-blue-800 cursor-pointer hover:bg-blue-50'  | ||||
|                         : 'text-gray-900' | ||||
|                     }`}
 | ||||
|                     onMouseEnter={(e) => handleItemsHover(event, e)} | ||||
|                     onMouseLeave={handleItemsLeave} | ||||
|                     onClick={(e) => handleItemsClick(event, e)} | ||||
|                     title={event.products.length > 0 ? 'Click to view items' : ''} | ||||
|                   > | ||||
|                     {event.products.length} item{event.products.length !== 1 ? 's' : ''} | ||||
|                     {event.products.length > 0 && ( | ||||
|                       <svg className="inline-block w-4 h-4 ml-1" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                         <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z" /> | ||||
|                       </svg> | ||||
|                     )} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap"> | ||||
|                     {event.total_amount ? ( | ||||
|                       <span className="text-sm font-semibold text-green-600"> | ||||
|                   <div className="text-right"> | ||||
|                     <p className="text-sm font-medium text-gray-900"> | ||||
|                       {new Date(event.date).toLocaleDateString()} | ||||
|                     </p> | ||||
|                     {event.total_amount && ( | ||||
|                       <p className="text-lg font-semibold text-green-600"> | ||||
|                         ${event.total_amount.toFixed(2)} | ||||
|                       </span> | ||||
|                     ) : ( | ||||
|                       <span className="text-sm text-gray-500">-</span> | ||||
|                       </p> | ||||
|                     )} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm text-gray-500"> | ||||
|                     {event.notes ? ( | ||||
|                       <span className="truncate max-w-xs block" title={event.notes}> | ||||
|                         {event.notes.length > 30 ? `${event.notes.substring(0, 30)}...` : event.notes} | ||||
|                       </span> | ||||
|                     ) : ( | ||||
|                       '-' | ||||
|                   </div> | ||||
|                 </div> | ||||
| 
 | ||||
|                 {event.products.length > 0 && ( | ||||
|                   <div className="mb-4"> | ||||
|                     <h4 className="text-sm font-medium text-gray-700 mb-2">Items Purchased:</h4> | ||||
|                     <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-2"> | ||||
|                       {event.products.map((product) => ( | ||||
|                         <div key={product.id} className="bg-gray-50 rounded px-3 py-2"> | ||||
|                           <div className="text-sm text-gray-900"> | ||||
|                             {product.name} {product.organic ? '🌱' : ''} | ||||
|                           </div> | ||||
|                           <div className="text-xs text-gray-600"> | ||||
|                             {product.amount} × ${product.price.toFixed(2)} = ${(product.amount * product.price).toFixed(2)} | ||||
|                           </div> | ||||
|                         </div> | ||||
|                       ))} | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 )} | ||||
| 
 | ||||
|                 {event.notes && ( | ||||
|                   <div className="mb-4"> | ||||
|                     <h4 className="text-sm font-medium text-gray-700 mb-1">Notes:</h4> | ||||
|                     <p className="text-sm text-gray-600">{event.notes}</p> | ||||
|                   </div> | ||||
|                 )} | ||||
| 
 | ||||
|                 <div className="flex justify-between items-center text-sm"> | ||||
|                   <div className="text-gray-500"> | ||||
|                     <div>Event #{event.id} • Created {new Date(event.created_at).toLocaleDateString()}</div> | ||||
|                     {event.updated_at && ( | ||||
|                       <div className="flex items-center mt-1"> | ||||
|                         <svg className="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||
|                           <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15" /> | ||||
|                         </svg> | ||||
|                         Updated {new Date(event.updated_at).toLocaleDateString()} | ||||
|                       </div> | ||||
|                     )} | ||||
|                   </td> | ||||
|                   <td className="px-6 py-4 whitespace-nowrap text-sm font-medium"> | ||||
|                   </div> | ||||
|                   <div className="flex space-x-2"> | ||||
|                     <button  | ||||
|                       onClick={() => handleEdit(event)} | ||||
|                       className="text-indigo-600 hover:text-indigo-900 mr-3" | ||||
|                       onClick={() => navigate(`/shopping-events/${event.id}/edit`)} | ||||
|                       className="text-indigo-600 hover:text-indigo-900" | ||||
|                     > | ||||
|                       Edit | ||||
|                     </button> | ||||
| @ -396,21 +159,14 @@ const ShoppingEventList: React.FC = () => { | ||||
|                     > | ||||
|                       Delete | ||||
|                     </button> | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|             </tbody> | ||||
|           </table> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             ))} | ||||
|           </div> | ||||
|         )} | ||||
|       </div> | ||||
| 
 | ||||
|       <AddShoppingEventModal | ||||
|         isOpen={isModalOpen} | ||||
|         onClose={handleCloseModal} | ||||
|         onEventAdded={handleEventAdded} | ||||
|         editEvent={editingEvent} | ||||
|       /> | ||||
| 
 | ||||
|       <ConfirmDeleteModal | ||||
|         isOpen={!!deletingEvent} | ||||
|         onClose={handleCloseDeleteModal} | ||||
| @ -419,54 +175,6 @@ const ShoppingEventList: React.FC = () => { | ||||
|         message={`Are you sure you want to delete this shopping event from ${deletingEvent?.shop.name}? This action cannot be undone.`} | ||||
|         isLoading={deleteLoading} | ||||
|       /> | ||||
| 
 | ||||
|       {/* Items Popup */} | ||||
|       {showItemsPopup && hoveredEvent && hoveredEvent.products.length > 0 && ( | ||||
|         <div  | ||||
|           className="items-popup fixed z-50 bg-white border border-gray-200 rounded-lg shadow-lg p-4 max-w-md" | ||||
|           style={{ | ||||
|             left: `${popupPosition.x + 10}px`, | ||||
|             top: `${popupPosition.y - 10}px`, | ||||
|             maxHeight: '300px', | ||||
|             overflowY: 'auto' | ||||
|           }} | ||||
|           onMouseEnter={() => setShowItemsPopup(true)} | ||||
|           onMouseLeave={handleItemsLeave} | ||||
|         > | ||||
|           <div className="space-y-2"> | ||||
|             {hoveredEvent.products.map((product, index) => ( | ||||
|               <div key={index} className="border-b border-gray-100 pb-2 last:border-b-0"> | ||||
|                 <div className="flex justify-between items-start"> | ||||
|                   <div className="flex-1"> | ||||
|                     <div className="text-sm font-medium text-gray-900"> | ||||
|                       {product.name} {product.organic ? '🌱' : ''} | ||||
|                     </div> | ||||
|                     <div className="text-xs text-gray-600"> | ||||
|                       {product.grocery?.category?.name || 'Unknown category'} | ||||
|                     </div> | ||||
|                     {product.brand && ( | ||||
|                       <div className="text-xs text-gray-500"> | ||||
|                         Brand: {product.brand.name} | ||||
|                       </div> | ||||
|                     )} | ||||
|                   </div> | ||||
|                   <div className="text-right ml-2"> | ||||
|                     <div className="text-sm font-medium text-gray-900"> | ||||
|                       ${product.price.toFixed(2)} | ||||
|                     </div> | ||||
|                     <div className="text-xs text-gray-500"> | ||||
|                       Qty: {product.amount} | ||||
|                     </div> | ||||
|                     <div className="text-xs font-medium text-green-600"> | ||||
|                       ${(product.amount * product.price).toFixed(2)} | ||||
|                     </div> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             ))} | ||||
|           </div> | ||||
|         </div> | ||||
|       )} | ||||
|     </div> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { Product, ProductCreate, Shop, ShopCreate, ShoppingEvent, ShoppingEventCreate, Brand, BrandCreate, Grocery, GroceryCreate, GroceryCategory, GroceryCategoryCreate, BrandInShop, BrandInShopCreate } from '../types'; | ||||
| import { Product, ProductCreate, Shop, ShopCreate, ShoppingEvent, ShoppingEventCreate, Brand, BrandCreate, Grocery, GroceryCreate, GroceryCategory, GroceryCategoryCreate } from '../types'; | ||||
| 
 | ||||
| // Use different API URLs based on environment
 | ||||
| const API_BASE_URL = process.env.NODE_ENV === 'production'  | ||||
| @ -54,16 +54,6 @@ export const brandApi = { | ||||
|   delete: (id: number) => api.delete(`/brands/${id}`), | ||||
| }; | ||||
| 
 | ||||
| // BrandInShop API functions
 | ||||
| export const brandInShopApi = { | ||||
|   getAll: () => api.get<BrandInShop[]>('/brands-in-shops/'), | ||||
|   getByShop: (shopId: number) => api.get<BrandInShop[]>(`/brands-in-shops/shop/${shopId}`), | ||||
|   getByBrand: (brandId: number) => api.get<BrandInShop[]>(`/brands-in-shops/brand/${brandId}`), | ||||
|   getById: (id: number) => api.get<BrandInShop>(`/brands-in-shops/${id}`), | ||||
|   create: (brandInShop: BrandInShopCreate) => api.post<BrandInShop>('/brands-in-shops/', brandInShop), | ||||
|   delete: (id: number) => api.delete(`/brands-in-shops/${id}`), | ||||
| }; | ||||
| 
 | ||||
| // Grocery Category API functions
 | ||||
| export const groceryCategoryApi = { | ||||
|   getAll: () => api.get<GroceryCategory[]>('/grocery-categories/'), | ||||
|  | ||||
| @ -122,19 +122,4 @@ export interface ShopStats { | ||||
|   total_spent: number; | ||||
|   visit_count: number; | ||||
|   avg_per_visit: number; | ||||
| } | ||||
| 
 | ||||
| export interface BrandInShop { | ||||
|   id: number; | ||||
|   shop_id: number; | ||||
|   brand_id: number; | ||||
|   created_at: string; | ||||
|   updated_at?: string; | ||||
|   shop: Shop; | ||||
|   brand: Brand; | ||||
| } | ||||
| 
 | ||||
| export interface BrandInShopCreate { | ||||
|   shop_id: number; | ||||
|   brand_id: number; | ||||
| }  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user