html-sketchapp-cli icon indicating copy to clipboard operation
html-sketchapp-cli copied to clipboard

Not able to perform consistent sync of Sketch Library File

Open lassediercks opened this issue 7 years ago • 23 comments

Hey there,

to be honest, I'm blown away how easily this turns my html into sketch symbols. Big props for that.

Let me explain what I'm struggling with:

  1. I turn a html file into the asketch.json file
  2. I trigger the sketch plugin and select both files
  3. I save that file
  4. I link the saved filed in the preferences as a library
  5. I create a new file and use the symbols from the library
  6. I update the html and perform step 1-3

I'd expect to have this image in my sketch but it does not appear. I'm able to use the updated symbols from the library but already used symbols are not updated.

Do you know this issue or how to fix it?

thanks for the awesome work <3

lassediercks avatar Feb 07 '18 13:02 lassediercks

Yeah, I was able to replicate this too. I'll have to find some time to dig into this a bit more and figure out what Sketch uses to decide whether a symbol is the same instance or not. In the meantime, if you have time to look into this too, that would be great. Once we've narrowed it down, it hopefully shouldn't be too hard to fix.

Also, thanks for the positive feedback! I really appreciate it 😄

markdalgleish avatar Feb 08 '18 02:02 markdalgleish

I had a look into this recently and realised it's quite tricky, and depends on whether or not html-sketchapp is able to generate consistent IDs. Any thoughts on this, @kdzwinel?

markdalgleish avatar Feb 26 '18 21:02 markdalgleish

Only IDs that come to my mind are objectIDs which ATM are randomly generated UUIDs. It should be possible to replace them with consistent ones. My guess is that Sketch only looks at the objectID of the symbol (and not its children), so that's the one I'd recommend playing with. I'd start by trying to set symbol._objectID (there is no public setter ATM) to something non-random. Not sure if it even has to be an UUID? Maybe any string will do? If so, symbol name should work.

I'd love to hear back if someone figures that one out!

kdzwinel avatar Feb 26 '18 22:02 kdzwinel

IIRC Sketch maintains a reference to both the symbol ID and the document ID.

markdalgleish avatar Feb 26 '18 22:02 markdalgleish

Ah, OK, but it looks like the document objectID is generated in the same way and can be replaced in the same way as for the symbol - document._objectID = '…'.

kdzwinel avatar Feb 26 '18 22:02 kdzwinel

I recently ran into this problem as well. I think fixing this is going to be pretty essential for maintaining a library over time. A couple thoughts:

  • Would it be possible to not include an objectID for the document in document.asketch.json at all? Given that the user is required to select an existing document in order to run the asketch2sketch plugin, it seems like the most reasonable thing would be to use the current objectID of that document. That would let the user choose whether to update an existing library or create a new one by selecting an existing document or creating a new one (pretty intuitive). IMO it's pretty surprising behavior to let the plugin change the identity of the document that the user has intentionally selected/created.
  • What would you think about simply hashing each symbol name to generate the objectID for the symbol? I've worked with Sketch objectIDs in the past and I'm pretty sure they don't have to be in any particular format, but either way it wouldn't be too hard to deterministically map something like an md5 hash into the same format as standard sketch objectIDs. The main downsides that I can think of are a) symbols with the same name would collide, potentially even across different libraries generated with html-sketchapp (I don't know how common duplicate names are IRL) and b) you couldn't update a symbol name without forking it (again, I don't know how common the scenario of wanting to update a symbol name and keep it synced with existing docs is)

elliotdickison avatar Mar 02 '18 17:03 elliotdickison

@kdzwinel not including a document ID in document.asketch.json makes sense to me. That threw me off for a while—I had to dig into the asketch2sketch source to realise that the document ID isn't used at all.

markdalgleish avatar Mar 05 '18 21:03 markdalgleish

@elliotdickison I'd say that hashing based on symbol name makes sense to me as a sensible default. In the long term, we could let you provide your own logic for generating IDs, which would allow you to provide backwards compatible hashes by aliasing symbol names, for example.

markdalgleish avatar Mar 05 '18 21:03 markdalgleish

we thought about it a bit and came up with this naming convention:

  • page gets a name of ui kit.
  • exported symbols get symbolID and do_objectID as a symbol name (Button/default)
  • underlying layers get do_objectID as an incremented version of a parent layer

*.asketch.json proposal with relevant keys included:

{
  "do_objectID": "nordnet-ui-kit",
  "layers": [
    {
      "name": "Badge/default",
      "symbolID": "Badge/default",
      "do_objectID": "Badge/default",
      "layers": [
        {
          "do_objectID": "Badge/default-0",
          "layers": [
            { "do_objectID": "Badge/default-0-0" }
          ],
        },
        {
          "do_objectID": "Badge/default-1",
          "layers": [],
        }
      ],
    },
    {
      "name": "Badge/danger",
      "symbolID": "Badge/danger",
      "do_objectID": "Badge/danger",
      "layers": [
        {
          "do_objectID": "Badge/danger-0",
          "layers": [
            { "do_objectID": "Badge/danger-0-0" }
          ],
        },
        {
          "do_objectID": "Badge/danger-1",
          "layers": [],
        }
      ],
    },
    {
      "name": "Badge/warning",
      "symbolID": "Badge/warning",
      "do_objectID": "Badge/warning",
      "layers": [
        {
          "do_objectID": "Badge/warning-0",
          "layers": [
            {
              "do_objectID": "Badge/warning-0-0",
              "layers": [],
            }
          ],
        },
        {
          "do_objectID": "Badge/warning-1",
          "layers": [],
        }
      ],
    },
    {
      "name": "Badge/success",
      "symbolID": "Badge/success",
      "do_objectID": "Badge/success",
      "layers": [
        {
          "do_objectID": "Badge/success-0",
          "layers": [
            {
              "do_objectID": "Badge/success-0-0",
              "layers": [],
            }
          ],
        },
        {
          "do_objectID": "Badge/success-1",
          "layers": [],
        }
      ],
    }
  ],
}

iamstarkov avatar Mar 08 '18 15:03 iamstarkov

@markdalgleish, @brainly, @lassediercks, @mlenser, @HerrBertling, @ronalson what do you think?

iamstarkov avatar Mar 08 '18 15:03 iamstarkov

I'd say if it works let's do it but from a technical point of view I cant judge the quality

lassediercks avatar Mar 08 '18 18:03 lassediercks

we will try it

iamstarkov avatar Mar 08 '18 19:03 iamstarkov

here we go https://github.com/brainly/html-sketchapp/pull/65

iamstarkov avatar Mar 08 '18 19:03 iamstarkov

and after that cli needs to be adjusted as well

iamstarkov avatar Mar 08 '18 19:03 iamstarkov

FYI html-sketchapp 3.2.0 has a proper ID setter on SymbolMaster now:

https://github.com/brainly/html-sketchapp/blob/cddaddadce4dcaeea02ec20755f0c2a3ff16eb0c/html2asketch/model/symbolMaster.js#L16-L18

@burakukula was able to successfully use it in Brainly style-guide import to have consistent symbols.

https://github.com/brainly/html-sketchapp-style-guide/blob/e4221b0e93582a4380f38f17c38045d0c7e5e5dd/src/styleguide2asketch.js#L49

kdzwinel avatar Jun 11 '18 08:06 kdzwinel

This is not a pressing issue for us internally. If anyone can offer some help implementing this, that would be much appreciated.

markdalgleish avatar Jun 21 '18 05:06 markdalgleish

I think #52 will fix at least some related parts of this. We've had problems where symbols imported to sketch library from asketch file cannot really be updated correctly as they lose for example text set for buttons.

#52 will allow to set symbol ID, which seems to make Sketch to handle component updates better and edited/added texts are not replaced with default texts (default = text in html).

We do still have some problems to have "Library update" available not appearing, but it is not critical to us as moving components for example a bit to the left and back to the right and then saving the file seems to trigger Sketch to notice update available.

mnikkane avatar Jun 21 '18 17:06 mnikkane

@mnikkane we haven't used library feature yet (although now that Sketch 51 supports text styles in libraries we might), so I haven't heard experienced the "Library update" issue you are describing yet. If there is something we can do in html-sketchapp to fix it - please open an issue for it.

We've had problems where symbols imported to sketch library from asketch file cannot really be updated correctly as they lose for example text set for buttons.

Instances of symbols loose overwritten text when master symbols are updated? We haven't experienced that 🤔Again, if there is antyhing we can do in html-sketchapp to fix it - let us know!

/cc @burakukula

kdzwinel avatar Jun 22 '18 08:06 kdzwinel

I think the problem in our case was that we have a Sketch library (generated completely with html-sketchapp-cli) and then design files, which use components (like buttons) from the library. And in design files button texts etc. are of course edited (texts overridden) according the needs in each design screen. But when we generated new library (with html-sketchapp-cli) to for example get updated button colors into use in Sketch library, the buttons in design files lost the edited texts and they were replaced with default texts defined in the library. I think we debugged this and noticed that the random id for symbols was the problem here.

But good news, the #52 seems to fix this update problem (when setting static id in symbolMasterMiddleware), so I think we're good after that!

mnikkane avatar Jun 22 '18 09:06 mnikkane

Hi, I've opened a PR in html-sketchapp to allow customising do_objectID for library sync. I found that symbol and text object IDs have to stay consistent for library updates.

/cc @kdzwinel

macintoshhelper avatar Nov 27 '18 11:11 macintoshhelper

@markdalgleish can you take a look at #67?

iamstarkov avatar Feb 12 '19 20:02 iamstarkov

@mnikkane the reason why Sketch is not showing the "update library" is because the changeIdentifier is always set to 0: https://github.com/brainly/html-sketchapp/blob/master/html2asketch/model/symbolMaster.js#L107. I'd recommend setting a integer generated from the Date instead.

That's what we do in react-sketchapp:

// Keep track on previous numbers that are generated
let previousNumber = 1;

// Will always produce a unique Number (Int) based on of the current date
function generateIdNumber() {
  let date = Date.now();

  if (date <= previousNumber) {
    previousNumber += 1;
    date = previousNumber;
  } else {
    previousNumber = date;
  }

  return date;
}

mathieudutour avatar May 13 '19 19:05 mathieudutour

@mathieudutour hmm, sounds interesting

iamstarkov avatar May 14 '19 01:05 iamstarkov